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优秀 的 软件 设计 简单 明了 。 不 过 很 遗憾 ， 如 今 的 计算 机 程序 Fick | 
Mine a, Bs A REMO AAA CG A GE Ze ee, AH 
简明 教程 则 在 帮助 读者 利用 科学 规则 掌 提 优秀 设计 的 基础 知识 ， 书 


中 给 出 的 甘 则 适用 于 所 有 编程 语言 和 软件 项 目 ， 并 且 永 还 有 效 。 


不 论 是 刚 人 门 的 程序 员 、 资 深 软件 工程 师 还 是 没有 技术 痛 景 的 管 
理 人 员 ， 读 过 本 书 之 后 ， 都 将 能 够 理解 如 何 创建 靠 谱 的 软件 项 目 计 


划 、 





确定 更 好 的 系统 模型 和 架构 。 


为 什么 软件 设计 成 了 一 门 缺失 的 科学 

软件 和 优秀 软件 设计 的 终极 目标 

确定 现在 以 及 将 来 软件 设计 的 价值 所 在 

用 真实 案例 证 明 系 统 如 何 随时 间 变 化 而 变化 

好 的 设计 能 适应 外 界 尽 可 能 多 的 变化 ， 而 软件 自身 的 变化 尽 可 能 少 
代码 越 简洁 ， 未 来 做 改动 的 难度 就 越 低 

测试 越 准确 ， 软 件 性 能 就 越 有 把 握 
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本 书 大 部 分 篇 幅 讲解 的 是 “软件 设计 "， 你 以 前 可 能 昕 过 这 个 i 
至 读 过 这 方面 的 书 。 不 过 ， 现 在 请 抛弃 原 有 的 观点 ， 从 全 新 的 、 准 确 
的 定义 开始 ， 重 新 认识 “软件 设计 ”。 


软件 是 什么 ， 我 们 都 知道 ， 因 此 真正 要 定义 的 束 是 设计 To 








动词 
(1) 为 创造 性 活动 制订 计划 。 例 句 :工程师 本 月 会 设计 一 座 桥梁 ， 下 
个 月 建造 它 。 


2 
) 为 某 种 创造 性 活动 而 制订 ， 但 尚未 实施 的 计划 。 例 句 ， 工程 师 已 

经 确定 了 桥梁 的 设计 ， 下 个 月 建造 它 。 

(2) 业已 存在 的 创造 物 所 遵循 的 计划 。 例 句 ， 那 座 桥 的 设计 相当 不 错 。 


谈 到 软件 设计 ， 下 列 这 些 定义 都 适用 。 


。 我 们 “设计 软件 ”( 动 词 “ 设 计 ") 时 ， 是 在 进行 计划 活动 。 设 计 软 
件 时 要 关心 的 事情 有 很 多 ， 包 括 代码 的 结构 、 所 用 的 技术 等 ， 还 要 
制订 许多 技术 决策 。 通 常 ， 我 们 只 是 在 脑子 里 做 这 些 决定 ， 有 时 候 
会 把 它 写 下 来 或 画 出 来 。 

。 上 一 步 的 结果 是 “软件 的 设计 ”( 第 一 种 意义 的 名 词 “ 设 计 ")， 也 
就 是 计划 。 同 样 ， 它 可 能 是 落实 下 来 的 文档 ， 也 可 能 是 我 们 脑 中 的 
若干 决定 。 

。 已 经 存在 的 程序 同样 有 “设计 ”( 第 二 种 意义 的 名 词 “ 设 计 ")， 也 
就 是 它 的 结构 ， 或 者 它 所 遵循 的 计划 。 有 些 程序 可 能 根本 没有 清楚 
的 结构 ， 所 以 是 “无 设计 ”的 ， 也 就 是 说 开发 它 的 程序 员 没有 任何 
明确 的 计划 。 在 “无 设计 ”和 “完整 的 设计 ”之 间 ， 存 在 着 广阔 的 
灰色 地 带 ， 比 如 “部 分 的 设计 ”.“ 在 某 段 程序 中 存在 的 若干 矛盾 的 
设计 ”、“ 接 近 完成 的 设计 ”等 等 。 一 些 刻意 而 为 的 糟粕 设计 甚至 比 
无 设计 还 要 差劲 。 比 如 你 遇 到 的 那些 刻意 制造 纠结 或 复杂 的 程序 ， 
就 属于 这 类 刻意 而 为 的 精 楼 设计 。 
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软件 设计 的 科学 就 是 为 软件 做 计划 、 制 定 决策 的 科学 ， 它 帮助 大 家 做 
出 这 类 决定 ; 


© 程序 的 代码 应 当 采 用 什么 结构 ? 
© 是 程序 的 速度 重要 ， 还 十 代码 容易 阅读 重要 ? 
© Ae tack, MA fla a? 


软件 设计 与 下 列 问 题 无 关 : 


© 公司 的 结构 应 该 十 怎 样 的 ? 

© 什么 时 候 召 开 团队 会 议 ? 

© 程序 员 的 工作 时 间 应 该 如 何 安排 ? 
， 程序 员 的 绩效 如 何 考核 ? 


这 些 决 策 与 软件 本 身 无 关 ， 只 与 组 织 有 关 。 显 然 ， 你 证 这 类 决策 的 全 
理性 也 是 重要 的 一 一 许多 软件 项 目 之 所 以 失败 ， 就 是 管理 失当 。 但 是 
这 不 是 本 书 的 主题 ， 本 书 关注 的 是 ， 如 何 为 你 的 软件 制订 合理 的 技术 
RR. 


ARR PETS RMA KRGRRAR, MATA ARA 
PPM RRAR, AT VSS] RGR 的 范畴 里 。 


21 程序 员 也 是 设计 师 

在 软件 项 目 中 ， 每 个 程序 员 的 工作 都 与 设计 有 关 。 首 席 程 序 员 人 负责 设 
计 程 序 的 总 体 架构 ， 高 级 程序 员 负 责 大 的 模块 ， 普 通 程 序 员 则 设计 自 
己 的 那 一 小 块 ， 其 至 只 是 某 个 文件 的 一 部 分 。 但 是 ， 即 便 仪 仅 是 写 一 
行 代码 ， 也 包含 设计 的 因素 。 


PTET, GPA. AN RARER Be Za, PU SRE. 
RAE ERT. ABM, PRATER, REA REESE 


每 个 写 代码 的 人 都 是 设计 师 ， 团 队 里 的 每 个 人 都 有 责任 保证 目 己 的 代 
码 有 着 民 好 的 设计 。 任 何 软件 项 目 里 ， 任 何 写 代码 的 人 ， 在 任何 层面 
上 ， 都 不 能 忽略 软件 设计 。 
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这 不 是 说 设计 应 该 采取 民主 程序 进行 。 设 计 决 不 应 该 由 某 个 委员 会 负 
责 ， 那 样 的 结果 必然 很 差劲 。 相 反 ， 所 有 的 开发 人 员 都 应 有 权 在 自己 
的 工作 中 做 出 良好 的 设计 决策 。 如 果 某 位 开发 人 员 做 了 糟糕 或 者 平庸 
的 决策 ， 资 深 开 发 人 员 或 者 首席 程序 员 就 应 当 推翻 这 些 决策 ， 重 新 来 
过 。 对 下 属 的 设计 ， 这 些 人 应 当 拥 有 否决 权 :。 不 过 ， 软 件 设计 的 责 
任 应 当 落 实在 真正 写 代码 的 人 身上 。 


身 为 设计 师 ， 必 须 时 时 愿意 聆听 建议 和 反馈 ， 因 为 程序 员 大 都 比较 陪 
明 ， 有 不 错 的 想法 。 但 是 考虑 了 所 有 这 些 建议 和 反馈 之 后 ， 任 何 决策 
都 必须 由 单独 的 个 人 而 不 是 一 群 人 来 做 出 。 


2.2 软件 设计 的 科学 
现在 ， 软 件 设 计 仍然 不 算 科 学 。 什 么 是 科学 ?词典 里 的 定义 7 


*。 科学 必须 包含 汇总 而 来 的 知识 。 也 就 是 ， 它 必须 包含 事实 而 不 是 意 
见 ， 且 这 些 事实 必须 汇总 起 来 (比如 集结 成 书 )。 

© 这 些 知 识 必 须 具 有 某 种 结构 。 知 识 必 须 能 分 类 ， 其 中 的 音 个 部 分 必 
须 能 够 依据 重要 性 之 类 的 指标 ， 妥 善 建立 起 与 其 他 部 分 的 联系 。 

*。 科学 必须 包括 一 般 性 的 事实 或 者 基本 的 规则 。 

© 科学 必须 告诉 你 在 现实 世界 中 如 何 司 一 些 事 情 。 它 必须 能 人 赵 应 用 到 
工作 或 生活 中 。 

*。 通常 ， 科 学 是 经 由 科学 方法 来 发 现 或 证 明 的 。 科 学 方法 必须 观察 现 
实 世界 ， 提 炼 出 关于 现实 世界 的 理论 ， 通 过 验证 理论 ， 而 且 这 些 实 
验 必须 是 可 重复 的 。 这 样 才能 说 明理 论 是 普 适 的 真理 ， 而 不 仅仅 是 
巧合 或 者 特例 。 






































注 1， 如 果 你 是 推 炙 决策 的 那个 人 ， 这 么 做 的 时 候 应 当 教 育 其 他 程序 员 。 你 应 当 说 明 ， 为 
什么 新 决策 比 原来 的 好 。 这 样 下 去 ， 你 以 后 推 都 的 决策 就 会 越 来 越 少 。 有 些 程序 员 
从 来 也 不 学 习 ， 即 便 是 这 样 教育 几 个 月 、 儿 年 ， 他 们 还 是 会 做 出 成 堆 的 业 梳 决策 , 
这 种 人 应 恋 清 理 出 团队 。 不 过 ， 大 多 数 程序 员 都 非常 聪明 ， 学 得 很 快 ， 所 以 这 基本 
不 是 一 个 问题 。 


SE 
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(KG, RINCASB TRS RMA. HARTA CRA 
述 ， 已 经 有 了 结构 。 不 过 ， 除 了 成 为 科学 所 必需 的 这 些 条 件 ， 我 们 还 
忽略 了 好 重要 的 部 分 : 清楚 表述 出 来 的 规则 (law)。 规 则 是 恒 常 不 变 
的 事实 ， 一 且 掌 握 了 它们 ， 人 就 不 会 犯错 。 


有 经 验 的 软件 开发 人 员 知 道 怎 么 做 才 是 正确 的 。 但 是 为 什么 这 么 做 就 
是 正确 的 ? 判断 正确 和 错误 的 标准 在 哪里 ? 


软件 设计 的 基础 规则 是 什么 ? 


本 书 将 给 出 答案 。 书 中 列 出 了 关于 软件 开发 的 若干 定 多 、 事 实 、 规 则 、 
定律 ， 它 们 的 着 眼 点 大 多 在 于 软件 设计 。 可 是 ， 定 多 (Definition), 
事实 (Fact), fi] (Rule), AUR (Law) 的 差别 在 哪里 昵 ? 


*。 定义 告诉 你 事物 是 什么 ， 应 当 如 何 使 用 。 

。 事实 是 关于 事物 的 真实 陈述 。 每 一 点 真实 的 信息 都 是 事实 。 

*， 条 例 是 给 你 的 确切 建议 ， 它 包含 基 些 具体 的 信息 ， 用 于 制订 决策 。 
但 是 ， 茶 例 并 不 能 帮 你 绝对 淮 确 地 预测 未 来 ， 也 不 能 帮 你 发 现 其 他 
真理 。 它 们 通常 会 告诉 你 是 否 需要 采取 某 些 行动 。 

。 规则 是 永远 为 真 的 事实 ， 它 涵盖 了 很 多 领域 的 知识 。 它 们 帮 你 发 现 
其 他 重要 的 真理 ， 帮 你 预测 未 来 要 发 生 的 事情 。 


所 有 这 些 里 面 ， 规 则 十 最 重要 的 。 阅 读 这 本 书 ， 你 会 清楚 地 认识 到 某 
些 规 由， 因为 书 里 会 明确 说 明 。 如 琳 你 不 是 很 确定 菜 些 信息 应 该 如 何 
归 类 ， 可 以 查看 附录 B， 其 中 列 出 了 本 书 中 所 有 重要 的 信息 ， 并 标注 
TEIRERN. AN, EX, Me HI, 


a PS EEE. AN, Al, SHE, FARSA: "Re: 
显然 的 ， 我 早 就 知道 了 。 这 扩 不 用 想 也 知道 ， 你 已 经 在 软件 开发 行 
业 里 摸 扑 党 打 很 人 人 了， 大 概 早 就 过 到 过 这 些 概 念 。 但 是 ， 你 这 么 想 的 
时 候 应 访问 问 目 己 : 


。 我 是 否 知道 ， 某 些 特定 的 说 法 是 否 经 过 了 证 实 ? 
。 我 是 否 清楚 它 的 重要 性 ? 
。 我 是 否 可 以 向 其 他 人 清楚 讲解 ， 让 对 方 彻底 明白 ? 
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部 分 知识 的 关系 如 何 ? 





*。 我 是 否 明 白 ， 在 软件 开发 领域 中 ， 它 与 其 


如 果 对 某 些 问 题 ， 你 曾经 说 过 “不 是 "、“ 或 者 "， 但 现在 可 以 说 
“是 "， 你 就 已 经 有 所 领悟。 要 区 分 某 些 说 法 到 底 是 科学 ， 或 者 仅仅 是 
想法 的 集合 ， 这 种 领悟 是 非常 重要 的 。 


当然 ， 科 学 不 等 于 全 知 全 能 。 字 宙 里 还 存在 待 发 现 的 秘密 ， 在 任何 领 
域 都 存在 着 未 知 的 事物 。 有 时 候 ， 发 现 了 新 的 数据 或 者 事实 ， 就 必须 
修正 某 些 基础 规则 。 但 这 正 是 新 的 出 发 点 ! 我 们 可 以 由 此 构筑 科学 : 
关于 开发 软件 的 、 切 合 实际 的 且 能 够 观察 验证 的 规则 和 真理 ， 


即便 在 某 天 ， 本 书 的 某 部 分 被 证 明 为 错 的 ， 并 出 现 了 更 完善 更 先进 的 
科学 ， 但 有 一 个 事实 是 很 清楚 的 : 软件 设计 可 以 成 为 科学 。 这 门 学 问 
不 是 什么 永恒 的 秘密 ， 因 为 程序 员 在 不 断 成 长 ， 容 询 顾问 要 把 软件 设 
计 新 方法 拿 出 去 换钱 。 
软件 设计 是 有 章 (规则 ) TAS, ENT MaDe, TH 

答 理 解 。 规 则 是 永恒 不 变 的 ， 是 基本 的 事实 ,而 且 确 实 可 行 。 
至 于 本 书 中 的 规则 是 不 是 正确 ， 我 们 可 以 引用 几 百 个 例子 和 试验 来 证 
明 它 ， 不 过 了 至 终 ， 你 需要 目 己 判断 。 请 检验 这 些 规 则 ， 看 看 你 是 否 
能 想到 关于 软件 开发 的 ， 适 用 范围 更 广 或 更 基础 、 更 深信 的 普 适 碳 
理 。 如 及 你 有 所 发 现 ， 或 者 找到 了 这 些 规则 的 问题 ， 可 以 登录 http:// 
codesimplicity.com/ 联系 作者 ， 珀 供 你 的 贡献 或 问题 。 关 于 这 | 上门 科学 
的 任何 进一步 发 展 ， 都 可 以 让 所 有 人 受益 ， 只 要 新 的 内 容 是 关于 软件 
设计 的 真实 的 、 基 本 的 定律 或 规则 。 


23 为 什么 不 存在 软件 设计 科学 
你 可 能 很 好 奇 ， 这 本 书 诞生 之 前 ， 为 什么 不 存在 软件 设计 的 科学 。 毕 
竟 ， 软 件 开发 已 经 有 几 十 年 历史 了 。 


让 我 来 讲 个 有 趣 的 故事 吧 。 这 些 背 景 知识 有 助 于 你 了 解 ， 为 什么 我 们 
跟 计 算 机 打 了 这 么 入 的 交道 ， 却 没有 建立 起 软件 设计 的 科学 ，。 
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备 ， 数 学 冢 想 用 机 器 代替 人 脑 来 解数 学 问题 。 


这 些 数学 家 才 是 计算 机 科学 之 父 ， 计 算 机 科学 正定 对 信息 处 理 所 做 
的 数学 研究 。 它 并 不 像 现 在 一 些 人 认为 的 那样 ， 龙 关于 计算 机 编程 
ma. 


不 过 ， 最 早 的 计算 机 是 在 计算 机 科学 家 的 指挥 下 、 由 经 验 直 富 的 电 
子 工 程 师 制造 出 来 的 ， 它 由 受过 专门 训练 的 操作 员 在 严格 控制 的 环 
境 下 操作 。 那 时 的 计算 机 是 为 有 需要 的 机 构 和 部 | (大 多 数 是 政府 
部 门 ， 为 了 计算 弹道 或 者 破解 密码 ) 量 身 定制 的 ， 每 种 型 号 只 生产 
一 两 台 设 备 。 


然后 UNIVAC 出 现 了 ， 这 是 世界 上 第 一 台 商 用 计算 机 。 当 时 ， 世 界 
上 还 有 足够 多 的 资深 的 理论 数学 家 。 到 了 普通 人 也 买 得 起 计算 机 的 时 
候 ， 已 经 不 可 能 为 每 台 计 算 机 配 一 各 数学 家 了。 有些 组 织 机 构 ， 比 如 
美国 统计 局 (也 就 是 UNIVAC 的 首 批 客户 之 一 )， 肯 定 是 有 足够 多 的 
受过 严格 训练 的 计算 机 操作 员 的 。 其 他 机 构 则 只 能 先 买 来 机 器 ， 然 后 
说 把 它 交 给 员工 :“ 好 ， 财 务 部 的 Bil， 这 机 器 归 你 了 ， 仔 细 看 看 操作 
手册 ， 让 它 跑 起 来 。 于 是 Bill 埋头 学 习 这 人 台 复 杂 的 机 器 ， 尽 可 能 让 
就 这 样 ，Bill 成 为 了 第 一 批 “ 应 用 程序 员 "。 他 可 能 在 学 校 学 过 数学 ， 
但 是 他 几乎 肯定 没有 学 习 过 设计 和 运行 计算 机 需要 用 到 的 高 深 理 论 。 
不 过 ， 他 可 以 阅读 手册 来 理解 ， 并 通过 试 错 法 让 计算 机 照 自 己 的 想法 
工作 ， 


然后 ， 商 用 计算 机 卖 得 越 来 越 多 ，Bill 这 样 的 人 也 越 来 越 多 ， 而 严格 
训练 的 操作 员 显 得 越 来 越 少 。 大 多 数 的 程序 员 就 像 Bill 一 样 。 如 果 说 
Bil 有 什么 不 一 样 的 地 方 ， 那 就 是 工作 压力 。 主 管 会 要 求 他 “完成 这 
个 任务 ”"， 而 且 会 说 “我 们 不 管 你 怎么 做 ， 总 之 去 做 就 是 了 ”。 他 必须 
根据 说 明 书 ， 想 象 出 该 如 何 完 成 任务 ， 然 后 完成 ， 即 便 他 的 程序 每 两 
个 小 时 崩溃 一 次 。 
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最 终 ，Bill 带领 一 组 同事 一 起 工作 。 他 必须 能 够 设计 出 系统 ， 并 且 将 
任务 分 派 给 不 同 的 人 。 于 是 ， 实 践 型 编程 的 技艺 不 断 发 展 完善 ， 这 个 
过 程 更 像 大 学 生 的 自学 ， 而 不 是 NASA 的 工程 师 建造 航天 飞机 。 


在 这 个 阶段 ， 软 件 开 发 就 像 一 锅 大 杂烩 ， 非 常 复杂 ， 也 难于 管理 ， 但 
是 ， 每 个 人 都 有 点 儿 心 得 。 然 后 出 现 了 Fred Brooks 写 的 《人 月 神 
话 少 ， 探 究 了 某 个 实际 的 软件 项 目的 开发 过 程 ， 并 指出 一 些 事实 。 其 
中 最 著名 的 就 是 ， 项 目 延 迟 的 情况 下 ， 增 加 更 多 的 程序 员 ， 只 能 加 剧 
延迟 。Brooks 并 没有 提出 完整 的 科学 ,但 是 他 对 编程 及 软件 开发 管理 
的 观察 相当 有 价值 。 


这 之 后 涌现 出 了 大 量 的 软件 开发 方法 : Rational 统一 过 程 、 能 力 成 熟 
度 模型 、 敏 捷 软 件 开发 ， 等 等 。 它 们 无 一 标榜 目 己 是 科学 一 一 虽然 它 
们 只 是 各 理 软件 开发 复杂 性 的 手段 。 


正 是 这 一 切 ， 导 致 了 现状 : 方法 众多 ， 真 正 的 科学 却 缺席 。 
其 实 ， 缺 席 的 科学 分 为 两 种 : 软件 管理 的 科学 ， 软 件 设 计 的 科学 。 


软件 管理 的 科学 告诉 我 们 的 是 ， 如 何 为 程序 员 分 派 工 作 ， 如 何 制 订 发 
布 计 划 ， 如 何 估量 任务 所 需 的 时 间 ， 诸 如 此 业 。 这 是 一 [] 显 学 ， 上 述 
的 各 种 方法 强调 的 正 是 这 一 点 。 在 这 个 领域 中 ， 存 在 着 彼此 矛盾 却 各 
有 道理 的 观点 ， 这 说 明 软 件 管理 的 基本 规律 尚未 被 认识 。 不 过 ， 大 家 
已 经 基 往 到 了 这 个 问题 。 

但 是 ， 软 件 设计 的 科学 在 现实 的 编程 中 却 设 有 什么 人 关 往 。 学 校 里 几 
乎 设 人 教导 你 说 ， 软 件 设计 中 存在 着 科学 。 相 反 ， 大 多 数 人 听 到 的 都 
Js: He ere, HERA S Cale, 
AVR APIA AMR TR o 

本 书 要 讲 的 并 不 是 计算 机 科学 ， 那 属于 数学 研究 。 本 书 提供 的 是 “ 实 
际 干 活 的 程序 员 ” 所 需 科 学 的 入 门 部 分 一 一 在 任何 语言 中 编写 程序 时 
都 应 当 遵循 的 者 干 基 础 定律 和 规则 。 这 种 科学 与 物理 和 化 学 一 样 可 
me, “CUM aS BF 
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据说 ， 这 样 的 科学 是 不 存在 的 ， 软 件 设 计 的 变化 太 多 了， 无 法 用 简 
单 、 基 础 的 规则 来 描述 ， 但 这 只 是 一 种 说 法 。 有 些 人 也 说 过 ， 理 解 物 
理 世界 是 不 可 能 的 ， 因 为 “世界 是 神 创造 的 ， 但 神 是 不 可 知 的 ， 可 
是 我 们 确实 已 经 发 现 了 物理 世界 的 规律 。 所 以 ， 除 非 你 相信 计算 机 是 
不 可 知 的 ， 否 则 ， 软 件 设 计 的 科学 是 必然 会 存在 的 。 


也 有 人 说 ， 编 程 纯 粹 是 一 门 三 术 ， 服 从 于 程序 员 的 个 人 爱好 。 这 么 说 
有 点 道 理 ， 在 科学 应 用 的 任何 领域 ， 都 存在 着 不 少 乙 术 的 成 分 ， 但 是 
它们 背后 必然 有 科学 存在 。 只 可 惜 ， 关 于 程序 设计 的 科学 至 今 仍然 是 
HH. 


其 实 ， 软 件 复杂 性 的 主要 根源 就 在 于 缺乏 科学 。 如 果 程 序 员 有 科学 指 
导 ， 知 道 该 如 何 开发 简单 的 软件 ， 就 不 会 有 这 么 多 的 复杂 性 问题 ， 我 
们 也 不 需要 运用 令 人 发 狂 的 过 程 来 管理 这 些 复杂 性 。 
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在 搞 写 程序 时 ， 我 们 应 当知 道 目 己 为 什么 要 编程 ， 芙 终 目标 是 什么 。 


有 没有 什么 办 法 ， 可 以 把 各 种 软件 的 目标 都 汇总 到 一 起 ?如 果 有 ， 那 
么 整个 软件 设计 的 科学 就 有 了 方向 ， 因 为 我 们 已 经 知道 自己 要 做 什么 。 


其 实 ， 全 部 软件 部 有 一 个 相同 的 目标 : 


帮助 其 他 人 。! 


依据 具体 的 软件 ， 我 们 可 以 得 到 更 加 具体 的 目标 。 比 如 ， 文 字 处 理 软 
帮助 大 家 编辑 文档 ， 浏 览 器 帮助 大 家 浏览 网 页 。 


有 些 软件 是 帮助 特定 人 群 的 。 比 如 ， 有 许多 审计 软件 给 审计 师 用 ， 此 


那么 ， 服 务 于 动物 或 植物 的 软件 呢 ? 软件 服务 动物 和 植物 的 真 
的 ， 还 是 帮助 人 类 。 


重要 的 是 ， 软 件 从 来 都 不 是 用 来 帮助 无 生命 事物 的 。 软 件 要 帮助 的 不 
是 计算 机 ， 而 是 入 。 即 便 你 写 的 是 程序 库 ， 它 们 也 是 为 程序 员 服务 
的 ， 程 序 员 也 是 人 ， 你 的 服务 对 象 并 不 是 计算 机 。 


“帮助 ”究竟 是 什么 意思 呢 ? 从 某 些 角度 来 看 ， 这 是 个 主观 的 概念 一 
对 这 个 人 有 帮助 ， 并 不 等 于 对 那个 人 也 有 帮助 。 不 过 词典 里 有 这 个 记 
的 定义 ， 所 以 它 的 定义 并 不 完全 依赖 于 个 人 的 理解 。《 韦 伯 斯 特 新 世 
界 美 语词 典 》 里 是 这 么 定义 “帮助 ”的 ， 


(GLA) 能 更 容易 地 做 某 事 ， 帮忙， 协助 。 特 指 完成 部 分 
IA, MESA DALIA E, 
你 可 以 帮 很 多 忙 一 一 做 规划 、 著 书 、 制 订 食 谱 等 。 想 干什么 取决 于 你 ， 
但 目标 都 是 提供 帮助 。 
























El: 这 个 目标 比 任何 规则 都 重要 。 在 英语 中 找 不 到 简单 的 词 来 横 述 它 ， 我 们 或 许可 以 称 
其 为 ”更 高 级 规则 ， 虽然 严 格 来 说 它 还 算 不 上 “规则 ”( 比 如 ， 它 不 能 陆 油 未 来 )。 
为 简单 起 见 ， 书 末 附 如 里 把 这 一 目标 归 类 为 规则 ， 其 他 时 候 ， 我 们 称 其 为 “软件 的 
目标 。 
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软件 的 目标 不 是 “赚钱 ”或 者 “炫耀 智商 "。 不 管 是 谁 ， 只 要 将 这 些 
当成 自己 的 目标 ， 就 偏离 了 软件 的 目标 ， 就 很 可 能 若 上 麻烦。 即便 赚 
钱 和 炫耀 智商 可 以 “帮助 ”你 ， 但 是 这 只 是 非常 狭隘 的 帮助 ， 相 比 为 
帮助 别人 而 设计 的 、 满 足 其 他 人 需求 的 软件 ， 仅 仅 考虑 狭隘 目标 的 设 
计 ， 很 可 能 收获 糟糕 的 软件 。? 

的 程序 ， 也 就 是 说 ， 


不 理解 “帮助 其 他 人 ”的 程序 员 ， 只 能 写 出 粳 
他 们 的 程序 提供 不 了 什么 帮助 。 实 际 上， 大概 存 在 这 么 一 点 理论 ( 根 
据 对 大 量程 序 员 的 长 时 间 观 察 所 得 的 猜想 ): 一 个 人 写 出 优秀 软件 的 
某 力 ， 完 全 取决 于 他 在 多 大 程度 上 理解 了 “帮助 其 他 人 ”的 思想 。 


总 的 来 说 ， 在 做 与 软件 有 关 的 决策 时 ， 指 导 法 则 就 是 判断 能 提供 什么 
样 的 帮助 请 记 住 ， 帮 助 有 很 多 种 ， 帮 大 忙 、 帮 小 忙 ， 帮 很 多 人 、 帮 
少数 人 )。 各 项 需求 也 可 以 照 这 个 标准 排出 先后 顺序 。 哪 项 需求 能 为 
人 们 提供 最 大 的 帮助 ， 就 应 当 赋 予 最 高 的 优先 级 。 关 于 需求 的 优先 
级 ， 还 有 很 多 可 以 补充 ， 但 是 在 评估 开发 或 维护 软件 系统 的 建议 时 ， 
“ 它 能 给 用 户 帮 多 少 忙 ”绝对 是 一 个 重要 而 且 基础 的 问题 。 


总 之 ， 在 设计 软件 时 ， 应 当 将 目标 一 一 帮助 他 人 一 一 视 为 应 该 考虑 的 
最 重要 因素 ， 这 样 ， 我 们 才能 认识 并 了 解 软件 设计 的 真正 科学 。 

















真实 应 用 


如 何 才能 把 软件 的 目标 落实 到 真正 的 项 目 中 去 呢 ? 举 个 例子 吧 ， 候 
设 我 们 要 为 程序 员 开发 一 款 文本 编辑 器 。 首 先 需 要 决定 的 就 是 软 


| 件 的 目的 。 这 个 目的 最 好 要 简章 ， 不 她 先 定 位 “帮助 程序 员 编 辑 文 
本 ”。 更 具体 一 点 也 没 问 题 ， 其 至 有 时 候 会 更 好 。 如 果 开 发 人 员 对 
更 具体 的 目标 沿 存 分 战 ， 且 从 最 莘 单 的 形式 开始 吧 。 





注 2: 没 错 ， 你 当然 可 以 把 “赚钱 ”当成 自己 或 者 公司 的 目标 ， 这 无 可 厚 非 ， 但 它 不 应 读 
古 你 的 软件 的 目标 。 任 何 情况 下 ， 你 所 赚 的 钱 都 直接 维系 于 你 的 软件 能 为 他 人 提供 
多 少 帮助 。 事 实 上 ， 决 定 软件 公司 收入 的 两 个 主要 因素 基本 就 是 组 织 水 平 (包括 行 
政 、 管 理 、 推 广 、 销售 等 方面 }， 以 及 软件 对 他 人 的 帮助 。 
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既然 目标 确定 了 ， 来 看 所 有 需要 完成 的 功能 吧 。 针 对 每 一 条 功能 ， 

我 们 都 要 问 自己 : “这 个 功能 怎样 帮助 程序 员 编 辑 文 本 呢 ? ”如 果 

答案 是 没什么 帮助 ， 功 能 就 应 当 马 上 了 取消。 这样 检查 过 一 过 之 后 ， 

对 剩 下 的 每 个 需求 ， 要 写 下 一 个 短 外 人 必 为 答案 。 假 设 ， 有 人 要 求 我 

们 为 常用 操作 提供 快捷 键 。 我 们 应 该 这 么 记 : “快捷 键 可 以 帮助 程 

序 员 编辑 文本 ， 因 为 有 了 快捷 键 ， 程 序 员 不 用 浪 您 时 间 输 入 ， 他 们 

与 程序 交互 的 达 度 提高 了 。” (你 不 一 定 要 写 下 来 ， 如 果 写 下 来 显 

得 多 余 ， 只 要 保证 自己 明 自 这 一 点 就 好 了 。) | 

之 所 以 要 这 样 问 自己 ， 是 因为 还 存在 其 他 有 价值 的 理由 。 

。 这 样 思 考 ， 有 助 于 澄清 功能 描述 或 实现 方式 中 的 模糊 之 处 。 上 
HAT RRA BEM, CO LMR, AAR 
价值 就 在 这 里 。 

。 这 样 思考 ， 有 助 于 团队 确认 功能 的 价值 。 有 些 人 可 能 不 喜欢 快 
捷 键 ,但 是 每 个 人 都 应 该 认同 上 面 关于 其 价值 的 解释 。 事 实 上 ， 
有 些 开 发 人 员 会 想 出 更 好 的 办 法 来 满足 客户 的 需求 (进一步 提 
高 操作 文本 编辑 器 的 速度 )， 吊 不 需要 快捷 键 ， 这 样 也 是 没有 
问题 的 。 如 果 问 题 的 答案 可 以 引出 更 好 的 想法 ， 就 应 读 去 和 实现 
那个 更 好 的 想法 。 问 题 的 葵 案 告诉 我 们 的 是 真实 的 需求 ， 而 不 
是 用 户 以 为 的 需求 。 

。 这 样 思考 ， 某 些 功能 会 凸显 出 更 重要 的 价值 。 这 有 助 于 项 目 侨 
导 分 配 优 先 级 。 

。 有 姬 一 万 步 说 ， 如 果 编 辑 器 因为 推倒 了 过 多 功能 而 脐 肿 ， 可 以 根 
据 这 些 葵 案 决 定 删 减 哪些 功能 。 

| 我 们 还 需要 列 一 张 bug 清 单 ， 方 便 检 查 和 反思 : 这 个 bug 如 何 影 响 程 

序 员 编辑 文本 ?有 了 时候 答案 很 明显 ， 那 么 也 不 需要 写 下 来 : 比如 你 

要 保存 文件 的 时 候 程序 前 清 了 ， 就 不 需要 画蛇添足 地 解释 为 什么 般 

清 是 不 好 的 。 

要 把 软件 的 目标 落实 到 真正 的 工作 中 ， 还 有 很 多 办 法 ， 这 里 只 给 出 

了 几 个 例子 ， 
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软件 设计 科学 的 目标 
了 解 了 软件 设计 的 目标 之 后 ， 就 可 以 确定 软件 设计 的 科学 的 基本 广 
加 I: 


从 目标 看 ， 我 们 知道 开发 软件 是 为 了 帮助 其 他 人 。 所 以 ， 软 件 设 计 科 
学 的 目标 应 该 是 : 


确保 软件 能 提供 尽 可 能 多 的 帮助 。 


其 次 ， 我 们 通常 希望 软件 可 以 给 大 家 提供 持续 的 帮助 。 所 以 ， 第 二 个 
目标 龙 ， 


确保 软件 能 持续 提供 尽 可 能 多 的 帮助 。 


这 个 目标 相当 伟大 ， 但 是 ,不 管 是 什么 软件 系统 ， 也 不 省 规模 多 大 ， 
它 都 是 非常 复杂 的 ， 所 以 确保 持续 提供 帮助 其 实 是 个 挑战 。 实 际 上 ， 
在 今天 ， 编 写 和 维护 有 帮助 的 软件 的 主要 障碍 在 于 设计 和 编程 。 如 果 
软件 很 难 开发 或 修改 ,程序 员 的 主要 精力 就 花 在 让 软件 “能 用 E, 
而 没有 精力 去 帮助 用 户 。 如 果 系 统 易于 修改 ,程序 员 就 会 有 更 多 的 余 
力 去 帮助 用 户 ， 不 必 费 心 于 编程 细节 。 同 样 道理 ， 软 件 的 维护 难度 越 
低 ， 程 序 员 确 保 软 件 能 持续 提供 帮助 的 难度 也 越 低 。 


于 古 ， 我 们 得 到 了 第 三 个 目标 : 


ALA ART HE RT RFORP RA RR, A 
的 系统 才能 为 用 户 提 供 尽 可 能 多 的 帮助 ， 而 且 能 持续 提供 尽 可 
能 多 的 帮助 。 
传统 上 认为 ， 这 第 三 个 目标 就 是 软件 设计 的 目标 ， 虽 然 从 来 也 没有 人 
这 么 清晰 地 表述 过 。 不 过 ， 第 一 个 和 第 二 个 目标 的 指引 是 非常 重要 
的 。 我 们 需要 记 住 ， 推 动 实现 第 三 个 目标 的 ， 就 是 确保 软件 现在 能 提 
供 帮 助 ， 将 来 仍然 能 帮助 。 


关于 第 三 个 目标 ， 需 要 特别 指出 的 是 “ 尽 可 能 简单 ”的 说 法 。 它 的 意 
思 是 软件 的 开发 和 维护 都 应 当 简 单 ， 要 避免 困难 或 复杂 。 这 并 不 是 
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说 ， 任 何事 从 一 开始 就 要 简单 ， 有 时 候 ， 学 习 新 技术 ， 做 一 份 好 的 设 
计 ， 都 要 花 很 长 时 间 ， 但 是 长 期 来 看 ， 你 做 出 的 选择 必须 确保 开发 和 
维护 软件 都 比较 而 单 。 


有 时 候 ， 第 一 个 目标 (确保 有 帮助 ) 和 第 三 个 目标 (维护 简单 ) 是 有 
点 冲突 的 ， 为 了 确保 有 帮助 ， 维 护 难度 就 会 增加 。 不 过 纵 观 历史 ， 这 
两 个 目标 冲突 的 原因 往往 并 不 是 如 此 。 开 发 出 完全 可 维护 、 对 用 户 极 
为 有 用 的 软件 ， 有 绝对 的 事实 依据 。 其 实 ， 如 果 你 不 能 确保 软件 的 可 
维护 性 ， 就 很 难 实现 第 二 个 目标 ， 也 就 不 能 持续 提供 帮助 。 这 样 看 
来 ， 第 三 个 目标 是 非常 重要 的 ， 否 则 我 们 就 没 办 法 完成 前 两 个 目标 。 
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软件 设计 师 面 对 的 主要 问题 是 :“ 在 设计 软件 时 ， 应 该 做 怎样 的 决 
定 ? ” 面 对 的 众多 可 能 ， 哪 一 个 才 是 最 好 的 。 我 们 不 是 要 确定 绝对 的 
好 坏 ， 而 是 要 知道 “这 些 可 能 的 选择 中 ， 哪 些 更 好 ? ”这 是 个 排序 
问题 ， 我 们 要 做 的 是 从 所 有 可 能 中 选 出 最 好 的 决定 。 举 个 例子 ， 设 计 
师 可 能 问 自 己 :“ 摆 在 眼前 的 功能 有 100 项 ， 但 我 们 的 人 力 只 能 够 完 
成 2 项 。 应 该 选 哪 2 项 呢 ? “ 


















度 ) 。 我 们 对 此 项 工作 的 


| 


V 表示 它 的 价值 。 该 变化 价值 几何 ?一 般 来 说 ,你 可 以 同日 已 ”这 个 
变化 对 用 户 有 多 少 用 "， 当 然 ， 还 有 很 多 其 他 方法 来 判断 其 价值 。 


所 以 ， 这 个 等 式 的 意思 是 : 


任何 一 点 改变 ， 其 合意 程度 与 其 价值 成 正比 ， 与 所 何 出 的 
成 本 成 反比 。 
这 并 不 是 在 判断 某 个 变化 绝对 对 错 ， 而 是 指导 你 如 何 分 辨 并 排序 你 的 
选项 。 能 市 来 较 太 价值 ， 屁 费 成 本 较 少 的 变化 ， 要 比 市 来 较 少 价 仁 ，、 


哪怕 你 的 问题 是 “我 们 是 否 应 当 维 持 不 变 ， 也 可 以 通过 这 个 方程 式 
得 到 答案 。 


















尔 可 以 间 自 己 “ 维 持 不 变 的 价值 何在 ”或 者 “维持 不 变 需 
要 花 什么 成 本 "， 再 对 比 进行 改变 的 价值 和 成 本 的 比率 。 
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计算 机 带 来 了 社会 的 剧变 。 因 为 有 了 它 ， 我 们 可 以 用 更 少 的 人 ， 干 更 
多 的 事情 。 这 就 是 计算 机 的 价值 一 一 它 可 以 干 很 多 的 事情 ， 而 且 速度 
相当 快 。 
这 挺 棒 的 。 
但 是 ， 计 算 机 会 出 问题 ， 而 且 总 出 问题 。 如 果 你 家 里 其 他 东西 出 问题 


有 计算 机 那么 频 瞎 ， 你 多 半 会 退货 。 生 活 在 现代 的 大 多 数 人 ， 每 天 至 
少 会 遇 到 一 次 系统 月 汗 或 者 程序 错误 。 


这 就 没 那么 棒 了 。 


1.1 计算 机 出 了 什么 问题 ? 


为 什么 计算 机 这 么 容易 出 问题 ?如果 是 软件 的 问题 ， 那 么 有 而 且 只 有 
一 个 原因 : 程序 写 得 太 糟糕 。 有 些 人 怪罪 管理 ， 有 些 人 怪罪 客户 ， 但 
是 调查 发 现 ， 问 题 的 根源 通常 都 在 于 编程 。 


那么 ,“ 程 序 写 得 太 糟 料 ” 是 什么 意思 ? 这 是 个 很 模糊 的 说 法 。 通 常 
来 说 ， 程 序 员 都 是 很 聪明 、 很 理智 的 人 ， 他 们 怎么 会 编 出 “和 精 糕 ”的 


程序 呢 ? 
说 穿 了 ， 这 一 切 都 与 复杂 性 有 关 。 


今天 ， 计 算 机 大 概 古 我 们 能 生产 的 最 复杂 的 设备 了 。 它 每 种 钟 可 以 计 
算数 十 亿 次 ， 它 内 部 数 以 亿 计 的 电子 元 件 必须 精密 协调 ， 整 台 计 算 机 
才 可 以 正常 运行 。 


计算 机 上 跑 着 的 程序 同样 复杂 。 举 例 来 说 ， 微 软 的 Windows 2000 还 
在 开发 时 ， 就 可 算 有 史 以 来 规模 最 大 的 软件 了 ， 它 包含 3000 万 行 代 
码 ， 这 大 概 相当 于 2 亿 字 一 一 是 《不 列 直 百科 全 书 》 字 数 的 5 售 。 


程序 的 复 灯 性 问题 可 能 更 加 用 烦 ， 因 为 程序 里 没有 措 得 着 的 东西 。 程 
序 出 问题 的 时 候 ， 你 也 找 不 到 什么 实 实在 在 的 东西 ， 打 开 瞧 瞧 里 面 发 
生 了 什么 。 程 序 完 全 是 抽象 的 ， 非 常 难处 理 。 其 实 ， 常 见 的 计算 机 程 



























4.1.1 价值 
方程 式 中 的 “价值 ” 指 什么 ? 价值 最 简单 的 定义 是 ， 

这 个 变化 能 带 给 人 多 大 帮助 。 
所 有 和 需 轨 帮 助 的 人 里 面 ， 最 重要 的 是 你 的 用 户 。 不 过 ， 为 了 赚 更 多 钱 
而 开发 某 些 功能 ， 也 是 一 种 形式 的 价值 一 一 对 你 来 说 有 价值 。 其 实 ， 
创造 价值 的 方式 有 很 多 种 ， 上 面 只 是 举 两 个 例子 。 
要 众 确 衡量 某 点 变化 所 创造 的 价值 ， 有 时 候 是 很 难得 。 如 果 你 的 软件 
可 以 帮 人 减肥 ， 你 怎么 精确 衡量 帮 人 减肥 的 价值 ? 这 是 做 不 到 的 。 不 
过 你 可 以 知道 的 是 ， 软 件 的 革 些 功能 可 以 帮 人 减 挥 不 少 体重 ， 有 些 功 
能 则 对 减肥 完全 无 效 。， 到 头 来， 你 还 是 可 以 根据 价值 来 为 变化 排序 。 
判 靳 [每 一 点 可 能 的 变化 的 价值 ， 基 本 的 依据 是 开发 人 员 的 开发 经 验 ， 
还 包 插 对 用 户 做 怡 当 的 研究 ， 找 到 对 他 们 帮助 最 大 的 工作 。 




















1. 价值 可 能 性 和 潜在 价值 
价值 由 两 部 分 组 成 : 可 能 价值 (这 个 变化 有 多 大 可 能 帮 到 用 户 )、 潮 在 
价值 (这 修 变化 在 对 用 户 提供 帮助 的 时 候 ， 将 为 用 户 提供 多 大 的 帮助 )。 


A= PAA IE: 


某 个 功能 可 以 延长 人 的 寿命 ， 即 便 只 有 百 万 分 之 一 的 可 能 性 ， 这 也 

仍然 是 非常 有 用 的 功能 。 它 的 潜在 价值 很 高 【延寿 )， 可 能 价值 非 
常 低 。 
再 举 个 例子 ， 你 可 以 为 电子 表格 程序 添加 功能 ， 以 便 言 人 输入 数 
字 。 盲 人 只 占用 户 的 很 小 一 部 分 ， 但 如 果 没 有 这 个 功能 ， 他 们 就 根 
本 没 法 使 用 你 的 程序 。 同 样 ， 这 个 功能 是 有 价值 的 ， 因 为 其 潜在 价 
值 很 高 ， 尽 管 它 只 对 一 小 部 分 用 户 有 用 【可 能 价值 很 低 )。 

。 能够 让 所 有 用 户 都 微笑 的 功能 确实 是 有 价值 的 。 其 潜在 价值 很 低 

笑 ) ， 但 是 它 能 影响 到 很 多 人 ， 所 以 它 的 可 能 价值 非 























© 反 过 来 ， 如 果 你 完成 的 某 项 功能 ， 只 有 百 万 分 之 一 的 可 能 性 让 某 
人 人 微笑， 那么 它 井 设 多 少 价 但 。 这 项 功能 的 蒂 企 价值 和 可 能 价值 
RIE. 

所 以 ， 在 判断 价值 时 ， 你 应 该 考虑 : 

© 多 少 用 万 〈 占 多 大 比例 ) 会 从 此 项 工作 中 受益 ? 

。 此 功能 对 用 户 有 价值 的 可 能 性 有 多 大 ? 或 者 换个 说 法 : 此 功能 发 挥 
价值 的 频率 有 多 高 ? 

*。 在 它 发 挥 价值 的 时 候 ， 它 能 发 挥 出 多 大 的 价值 ? 


2. 平衡 危害 
有 些 改变 在 带 来 帮助 的 同时 也 会 带 来 麻烦 。 如 果 你 的 程序 要 展示 广 
告 ， 就 会 让 某 些 人 觉得 烦 ， 虽 然 展示 广告 可 以 为 开发 提供 资金 。 


权衡 改变 的 价值 ， 需 要 衡量 它 可 能 造成 的 危害 ， 并 权衡 利弊 。 


3. mate PRIME 
如 果 某 个 功能 找 不 到 用 尸 ， 就 不 存在 实际 的 价值 。 这 些 功能 可 能 用 户 
找 不 到 ， 或 者 很 难 使 用 ， 或 者 根本 就 帮 不 上 任何 人 。 在 将 来 它们 可 能 
有 价值 ， 但 目前 没有 。 


这 也 意味 着 ， 大 多 数 情 况 下， 为 了 确保 你 的 软件 有 价值 ， 就 必须 真正 
发 布 它 。 如 果菜 个 变化 花费 的 时 间 太 长 ， 最 后 可 能 就 没有 任何 价值 ， 
因为 它 不 能 及 时 发 布 ， 无 法 给 人 提供 切实 的 帮助 。 在 判断 菏 项 工作 的 
侣 痘 程 度 时 ， 评 佑 其 发 布 计 划 是 非常 重要 的 。 


41.2 成 本 

比 起 价值 ， 成 本 更 容易 量化 一 点 。 通 常 ， 你 可 以 计算 “需要 多 少 个 人 工 
作 多 少 小 时 ”。 经 常 听 到 的 关于 成 本 的 量化 表述 是 “100 ASE”, ‘ERAN 
1 个 人 工作 1004, 或 者 100 个 人 工作 1 年 ， 或 者 S0 个 人 工作 2 年 。 


尽管 成 本 可 以 量化 ， 落 实 起 来 却 相 当 复杂 ， 甚 至 是 不 可 能 的 。 有 些 变 
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(eee, WWM, Cee EFA RRM EE HN bug 
上 的 时 间 就 很 难 预测。 不 过 ， 经 验 丰富 的 软件 开发 人 员 仍然 可 以 在 不 
需要 知道 确切 数值 的 情况 下 ， 根 据 可 能 成 本 来 预 而 。 


预 铀 某 个 变化 的 成 本 时 ， 重 要 的 是 要 考虑 牵涉 到 的 所 有 投入 ， 而 不 仅 
仅 是 编程 的 时 间 。 研 究 要 花 多 少时 间 ? 开发 人 员 之 间 的 沟通 要 花 多 少 
时 间 ? 所 做 的 思考 要 伦 多 少时 间 ? 


简单 说 ， 与 这 个 变化 有 关 的 每 一 点 时 间 ， 都 是 成 本 的 一 部 分 。 


4.1.3 ER 

到 目前 为 止 ， 这 个 方程 式 还 非常 简单 ， 但 它 忽略 了 一 个 重要 的 因 
索 一 一 时 间 。 你 不 但 需要 完成 这 个 变化 ， 还 需要 一 直 维 护 它 。 所 有 的 
变化 必需 要 维护 。 有 些 工 作 显 而 易 见 机 做 维护 :， MRA RRR 
报税 的 软件 ， 每 年 的 新 规定 公布 之 后 ， 你 都 必须 升级 。 即 使 这 项 工作 
暂时 看 不 到 长 期 的 维护 成 本 ， 即 使 维护 成 本 只 是 你 下 一 年 必须 再 做 一 
次 测试 ， 维 护 成 本 仍然 存在 。 


我 们 还 必须 考虑 现在 价值 和 未 来 价值 。 修 改 现 有 的 系统 时 ， 受 益 的 是 
现在 的 用 户 ， 但 是 它 也 可 能 帮助 到 未 来 的 用 户 ， 甚 至 可 能 影响 到 未 来 
的 用 户 数目 ， 进 而 影响 到 现 有 系统 究竟 总 共 能 帮助 到 多 少 人 。 
有 些 功 能 的 价值 会 随时 间 变 化 。 举 例 来 说 ，2009 年 的 税收 计算 软件 在 
2009 年 和 2010 年 是 有 价值 的 ， 但 到 了 2011 年 就 要 打折 扣 了 。 这 就 是 
随时 间 变 化 而 贬值 的 功能 。 还 有 些 功 能 会 随时 间 变 化 而 增值 。 
所 以 ， 脚 踏实 地 来 看 这 个 问题 ， 我 们 会 发 现 ， 成 本 包含 实现 成 本 和 维 
护 成 本 ， 价 值 也 包括 当前 价值 和 未 来 价值 。 用 方程 式 来 表示 就 是 ; 
E=E,+E, 
= 六 十 了 区 





其 中 ; 
E 代表 实现 成 本 
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En 代表 维护 成 本 
V, 代 表 当 前 价值 
广 代 表 未 来 价值 


4.1.4 ”完整 的 方程 式 
考虑 所 有 因素 ， 完 整 的 方程 式 就 是 : 





用 文字 说 明 束 是 : 


改变 的 合意 程度 (可行 性 ) ， 正 比 于 软件 当前 价值 与 未 来 


价值 之 和 ， 凤 比 于 实现 成 本 和 维护 成 本 之 和 ， 


这 就 是 软件 设计 的 最 重要 规律 。 不 过 ， 还 需要 做 一 些 补充 说 明 。 


415 化 简 方 程式 
oy ene 


















工作 能 带 来 多 少 钱 ,“ 成 本 ”衡量 的 是 为 完成 这 项 工作 需要 投入 多 少 
钱 。 在 真实 世界 中 不 应 当 这 样 处理 ， 但 就 这 个 例子 而 言 ， 这 么 做 可 以 
起 到 简化 作用 。 
入 ， 假 设 要 完成 工作 对 应 的 方程 式 是 这 样 ; 
_ $10 000 + $10 000/ 天 

$1000 + $100/ 天 
这 项 工作 要 花 1000 美元 完成 (实现 成 本 ， 分 母 的 第 1 项 )， 能 马上 带 
来 10 000 美元 的 收益 (当前 价值 ， 分 子 的 第 1 项 )。 之 后 每 一 天 ， 它 
都 能 带 来 1000 美元 收益 (未 来 价值 ， 分 子 的 第 2 项 )， 而 且 需 要 化 
100 美元 来 维护 (维护 成 本 ， 分 盘 的 第 2 项) 


到 第 10 天 ， 累 积 的 未 来 价值 是 10 000 美元 ， 昧 积 的 维护 成 本 是 1000 




















是 时 间 在 10 





美元 。 这 也 相当 于 之 前 说 的 “当前 价值 ”和 维护 成 本 ， 只 : 
天 后 而 已 。 


在 100 天 后 ， 未 来 价值 一 共 是 100 000 美元 ， 维 护 成 本 是 10 000 美元 。 


在 1000 天 后 ， 未 来 收益 总 计 为 100 万 美元 ， 维 护 成 本 总 计 为 10 万 
美元 。 这 时 候 ， 节 开始 说 的 “当前 价值 ”和 维护 成 本 看 来 就 微不足道 
了 。 随 着 时 间 的 访 逝 ， 它 会 越 来 越 不 重要 ， 甚 至 完全 无 足 轻 重 。 于 





Es HEN, ATTERERT : 
_ YU 
D=— 


其 实 ， 几 乎 所 有 软件 设计 的 决策 都 完全 忽略 了 未 来 价值 与 维护 成 本 的 
对 比 。 有 时 候 当 前 价值 和 实现 成 本 足够 大 ， 始 终 在 决策 中 占有 决定 性 
地 位 ， 但 是 这 种 情况 很 少见 。 一 般 来 说 ， 软 件 系统 都 需要 维护 很 长 时 
间 ， 大 多 数 情 况 下 ， 未 来 长 期 收益 和 维护 成 本 才 是 真正 需要 考虑 的 ， 
与 之 相 比 ， 当 前 价值 和 实现 成 本 变 得 无 足 轻重 。 


4116 ”你 需要 什么 ， 不 需要 什么 
我 们 已 经 学 到 了 重要 的 一 课 ， 也 就 是 要 避免 这 样 的 情况 : 对 某 项 工 


作 ， 维 护 成 本 最 终 大 大 超过 未 来 的 收益 。 假 设 你 完成 某 项 工作 , 在 5 
天 内 ， 它 的 成 本 和 价值 看 起 来 是 这 样 的 ， 


天 数 成 本 价值 


1 $10 $1000 
$100 $100 

3 $1000 $10 

4 $10 000 $1 

5 $100 000 $0.10 

it $111 110 $1111.10 


A, AE TNE EER IAE, TROPA. MRM TH 


El: HERE: MAFIA. ARES, EAS AA EY RL 
CAMARA. EME, PACO FATE ERA RRA 
Sr, MAX TRADE. Fa, AE, HEN IBAARIE, 
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头 持续 下 去 ， 系 统 根本 无 法 维护 一 一 它 会 变 得 无 比 昂 贯 ， 而 每 天 都 设 
有 产生 收益 。 


无 论 如 何 ， 只 要 维护 成 本 的 增长 速度 比价 值 快 ， 你 就 会 进 到 贱 烦 ， 即 
便 刚 开始 的 情况 是 这 样 的 ; 


1 $1000 $1000 
2 $2000 $2000 
3 $4000 $3000 
4 $8000 $4000 
总 计 $15 000 $10 000 


ER th BI GE eh AME ik BEE ORIEL RAY 
系统 ， 保 证 维护 成 本 随时 间 降 低 ， 最 终 降 到 零 (或 者 尽 可 能 接近 零 )。 
只 要 你 能 做 到 这 点 ， 就 无 所 谓 未 来 收益 是 大 还 是 小 ， 总 之 你 不 需要 关 
心 。 比 如 ， 下 表 所 示 就 是 合意 的 情况 。 


天 数 成 本 价值 

1 $1000 $0 

$100 $10 

$10 $100 

$0 $1000 
$0 $10 000 
$1110 $11 110 


PN 
quel 
ops 


价值 
$10 
$10 $10 
$5 $10 
$1 $10 
5 $0 $10 
总 计 $36 $50 


an 
E 

SE 

5 > 


能 带 来 更 高 未 来 价值 的 改变 仍然 是 更 可 取 的 ， 不 过 ， 只 要 每 项 决策 
的 维护 成 本 都 会 随时 间 降 低 到 接近 于 零 ， 未 来 你 就 不 会 陷 人 危险 的 
境地 。 
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BML, ARARK Re TEP RE, IHRER FEN. Ar 
以 ， 哪 怕 维 护 成 本 和 未 来 收益 都 增加 ， 只 要 未 来 收益 超过 维护 成 本 ， 
也 是 值得 做 的 。 


天 数 成 本 价值 


1 $1 $0 
2 $2 $2 
3 $3 $4 
4 $4 $6 
5 $5 $8 
总 计 $15 $20 


这 样 的 工作 也 不 错 ， 但 是 更 合意 的 是 这 样 的 工作 :虽然 实现 成 本 很 
高 ， 维 护 成 本 却 会 下 降 。 如 果 维护 成 本 会 随时 间 推移 而 逐步 降低 ， 这 
IRRE, 所 以 我 们 才 说 这 样 做 更 合意 。 


通常 来 说 ， 如 果 要 设计 一 个 系统 ， 其 维护 成 本 能 逐渐 降低 ， 需 要 在 实 
现 上 花 很 高 的 成 本 ， 做 相当 多 的 设计 和 规划 。 不 过 请 记 住 ， 在 设计 决 
策 中 ， 实 现成 本 通常 并 不 是 重要 的 因素 ， 所 以 基本 可 以 忽略 。 
简 而 言 之 

相 比 降 做 实现 成 本 ， 降 做 维护 成 本 有 更 加 重要 。 
在 软件 设计 的 科学 中 ， 这 是 非常 重要 的 一 点 ， 
但 是 维护 成 本 从 哪里 来 ?怎样 设计 系统 ， 才 能 保证 维护 成 本 会 随时 间 
推移 而 降低 ? 本 书 剩 下 的 篇 幅 ， 大 部 分 讲 的 都 是 这 个 主题 。 但 是 在 开 
始 之 前 ， 先 要 对 未 来 做 更 细致 的 检查 。 


42 ”设计 的 质量 

如 今 ， 要 写 出 一 款 能 帮助 某 沾 人 的 软件 ， 实 在 是 太 容 易 了 。 但 是 ， 要 
写 出 能 帮助 几 百 万 人 ， 而 且 是 在 未 来 几 十 年 里 持续 提供 这 种 帮助 的 软 
件 ， 则 要 难得 多 。 那 各 ， 编 程 中 的 大 部 分 精力 应 该 花 在 哪里 ? KER 
AS ETT Ali eB FARE? 现在 ， 还 是 未 来 的 几 十 年 ? 
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答案 是 ， 相 比 现在 ， 将 来 有 多 得 多 的 编程 工作 要 做 ， 也 有 更 多 的 用 户 
要 帮助 。 在 未 来 ， 你 的 软件 必须 保持 部 和 争 力 ， 才 能 继续 存在 ， 同 时 ， 
稚 护 成 本 和 用 户 数 量 也 会 增加 。 


如 果 我 们 忽视 事实 ， 放 弃 对 未 来 的 思考 ， 只 考虑 当下 “能 用 ”的 软 
件 ， 那 么 我 们 的 软件 在 未 来 就 会 很 难 维护 。 如 果 软 件 很 难 维护 ， 就 很 
难 确保 它 能 够 帮助 别人 (而 这 正 是 软件 设计 的 目标 )。 如 果 你 不 能 添 
加 新 功能 ， 也 不 能 修正 问题 ， 你 最 后 得 到 的 就 是 “ 精 糕 的 软件 "。 它 
不 能 帮 到 用 户 ， 而 且 满 身 问题 。 
于 是 ， 可 以 得 到 这 条 规律 ， 

设计 的 质量 好 坏 ， 正 比 于 该 系统 在 未 来 能 持续 帮助 他 人 时 

间 的 长 度 。 

如 果 你 的 软件 只 能 在 未 来 几 小 时 内 提供 帮助 ， 就 不 需要 花 太 多 的 工夫 
去 设计 。 如 果 需 要 在 未 来 10 年 内 都 能 派 上 用 场 (这 个 时 间 通 常会 超过 
你 的 预期 ， 虽 然 你 很 可 能 只 打算 保证 能 用 6 个 月 ) ， 就 需要 花 很 多 精力 
去 设计 。 如 果 举 棋 不 定 ， 不 妨 设 想 它 要 使 用 很 长 很 长 的 时 间 ， 然 后 按 
照 这 个 设想 去 设计 。 不 要 把 自己 禁 铀 在 某 种 工作 定 势 里 ， 要 保持 灵活 ， 
不 要 做 任何 以 后 无 法 改变 的 决策 ， 在 设计 时 要 慎重 ,慎重 ， 再 慎重 。 


4.3 不 可 预测 的 结果 
现在 我 们 知道 了 ， 设 计 软 件 时 最 应 该 关注 的 是 未 来 。 不 过 ， 关 于 任何 
工程 ， 都 有 一 点 极为 重要 : 
未 来 的 某 些 事情 ， 是 我 们 所 不 知道 的 。 | 
其 实 ， 软 件 设 计 也 是 如 此 ， 关 于 未 来 ， 大 多 数 事情 都 是 未 知 的 。 


程序 员 犯 的 最 常见 也 是 晤 严重 的 错误 ， 就 是 在 其 实 不 知道 
未 来 的 时 候 去 预测 未 来 。 


假设 ，1985 年 有 个 程序 员 编 了 个 修理 软盘 错误 的 程序 。 这 个 程序 不 能 
修理 其 他 任何 东西 一 一 因为 它 的 每 个 部 分 都 是 专门 针对 软盘 的 工作 原 
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理 和 状态 而 考 虚 的 。 到 现在 ， 这 个 软件 肯定 没 用 了 ， 因 为 已 经 没有 人 
用 软盘 了。 程序 员 的 预期 是 “人 们 会 一 直 使 用 软盘 ”一 一 但 其 实 他 并 
不 确切 知道 这 一 所 。 

预测 短期 未 来 是 可 能 的 ， 但 长 期 未 来 基本 是 未 知 的 。 可 是 ， 相 比 短 


期 ， 长 期 的 未 来 对 我 们 来 说 更 重要 ， 因 为 我 们 的 设计 决策 会 在 未 来 更 
长 的 时 间 里 产生 更 大 的 影响 。 





如 果 完 全 不 考虑 未 来 ， 只 根据 当前 已 知 的 确切 信息 确定 所 
AKHAR, PRMADPARS So 








这 个 说 法 听 起 来 与 本 章 之 前 说 的 相 秆 盾 ， 其 实 并 非 如 此 。 在 进行 决策 
时 ， 未 来 才 是 最 重要 的 事情 。 但 在 进行 决策 时 ， 考 虑 未 来 的 变数 和 党 
试 预测 未 来 ， 是 有 区 别 的 。 


打 个 比方 ， 你 要 做 个 简单 的 抉择 ， 是 吃 东西 还 是 饿 死 。 这 并 不 需要 预 
测 未 来 才 可 以 雇 东 -你 当然 知道 吃 乐 西 是 更 好 的 雇 课 。 为 什么 ? Al 
为 这 样 你 可 以 话 下 去 ， 而 且 将 来 可 以 过 得 更 好 ， 饿 死 了 就 设 有 未 来 可 
言 了 。 未 来 是 重要 的 ， 在 做 决策 时 需要 考虑 它 。 选 择 吃饭 ， 是 因为 它 
可 以 带 来 更 好 的 未 来 。 但 是 ， 未 来 并 不 需要 预 铀 一 一 我 们 不 需要 知 起 
确切 的 未 来 ， 比 如 “我 要 吃 东 西 ， 因 为 明天 我 得 救 一 个 护 了 于 。 只 要 
你 选择 了 吃饭， 而 不 选择 饿 死 ， 无 论 明 天 发 生 什 么 ， 部 比 饿 死 要 好 。 


同样 道理 ， 在 软件 设计 时 ， 可 以 根据 已 知 的 信息 做 茶 些 诀 提 ， 目 的 是 
为 了 创造 更 好 的 未 来 (提升 价值 ， 降 低 维护 成 本 )， 而 不 必 巴 测 未 来 
究竟 会 发 生 什 么 具体 的 事情 。 


当然 ， 也 有 少量 例外 ， 有 了 时候 你 确切 知道 未 来 短期 内 会 发 生 什 么 ， 便 
可 以 据 此 决策 。 如 果 这 么 做 ， 必 须 对 未 来 有 相当 的 把 握 ， 而 且 这 种 未 
来 必须 触手 可 及 。 无 论 你 名 脆 明 ， 肯 定 没有 简单 的 办 法 惟 确 预 测 长 期 
未 来 。 


举 个 编程 之 外 的 例子 。CD 诞生 于 1979 年 ， 基 初 设想 是 取代 磁带 ， 成 
为 最 主要 的 听 音 媒质 。 谁 能 预计 到 20 年 之 后 会 出 现 同样 尺寸 的 DVD, 
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所 以 计算 机 上 会 出 现 同 时 兼容 CD/DVD 的 驱动 器 ? 谁 又 可 以 预计 到 ， 
当 CD-ROM 以 50 倍速 读 取 CD 时 ， 会 出 现 什 么 问题 ? 


正 是 因为 如 此 ， 任 何 工程 一 包括 软件 开发 一 都 有 “指导 原则 ”"。 如 
果 我 们 遵循 这 些 原则 ， 无 论 未 来 发 生 什么 ， 一 切 事情 都 会 按部就班 。 
这 就 是 软件 设计 的 规律 和 法 则 ， 也 是 我 们 这 些 设计 师 的 “指导 原则 ”。 


是 的 ， 重 要 的 是 要 记 住 ， 存 在 着 未 来 。 但 是 ， 这 不 要 求 你 必须 预测 未 
来 ， 它 只 是 说 明 你 为 什么 应 当 遵循 本 书 的 规则 和 条 例 来 决策 一 一 因为 
无 论 未 来 会 发 生 什 么 ， 它 们 总 可 以 保证 你 的 软件 不 偏离 正轨 ，。 


事实 上 ， 某 条 特定 的 规则 和 条 例 在 未 来 会 以 什么 样 的 方式 帮助 你 也 无 
法 预测 一 一 但 是 它 确实 能 帮 上 忙 ， 而 且 你 会 欣然 乐意 把 它们 应 用 在 工 
作 中 。 


你 当然 有 权 不 同意 本 书 中 读 到 的 规则 。 关 于 规则 ， 你 一 定 要 有 自己 的 
看 法 。 但 是 你 应 当 谨 记 ， 如 有 果 不 遵 守 这 些 规则 ， 你 很 可 能 在 无 法 预 宙 
的 未 来 落得 一 团 精 。 
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序 就 已 经 足够 复杂 ， 没有 人 可 以 从 头 到 尾 理解 所 有 代码 十 如 何 工作 
的 。 程 序 越 大 ， 越 是 如 此 。 


这 样 说 来 ， 编 程 就 成 了 把 复杂 问题 化 解 为 简单 问题 的 灾 动 。 否 则 ， 一 
日 程序 达到 菜 种 复杂 程度 ， 就 没有 人 可 以 理解 了 。 程 序 中 复 灯 的 部 分 
必须 以 某 种 简单 方式 组 织 起 来 ， 这 样 ， 不 需要 神 那 样 强大 的 思维 ， 普 
通 程 序 员 也 可 以 开发 出 来 。 


AEREA E A A LARA EMBA. 


ZERO” ENSERRHMN. (ELA Ra ARMA 
来 就 够 复杂 了 ) Sd “RED” MET, BRO AIC ARTE, Mi 
有 考虑 降低 其 他 程序 员 需 要 面 对 的 复杂 性 。 


大 概 就 是 这 么 一 回 事 。 


设想 一 下 ， 为 了 把 钉子 钉 到 地 板 上 ， 工 程 师 发 明了 一 合 机 得 ， 上 面 有 
BRE, BET, DAM. Mirta Roda. 


再 设想 ， 有 人 告诉 你 说 :“ 我 需要 一 段 代 码 ， 它 能 用 在 任何 程序 的 任 
何 地 方 ， 能 够 通过 任何 介质 ， 实 现 两 台 计 算 机 之 间 的 通信 。 这 个 问 
ATC RETR MECH Atel. ATLA, AERA (KSB A) 在 这 种 情 
况 下 给 出 的 解法 ， 就 像 是 一 台 装 备 了 皮带 轮 、 绳 子 、 磁 铁 的 机 器 ， 其 
他 人 当然 很 难看 得 懂 。 并 不 是 这 些 程序 员 缺 少 理性 ， 他 们 的 脑子 也 设 
有 进 水 。 他 们 面 对 的 问题 确实 困难 ， 设 定 的 期 眼 也 很 暴 ， 他 们 能 做 的 
就 是 这 些 。 在 这 些 程 序 员 看 来 ， 写 出 来 的 东西 是 能 用 的 ， 它 符合 要 
求 。 这 就 是 他 们 的 老板 需要 的 ， 看 来 也 应 该 是 客户 需要 的 。 


不 过 ， 这 些 程序 员 毕 竟 没 做 到 化 党 为 向 。 完 工 之 后 ， 他 们 把 结果 交 给 
其 他 程序 员 ， 其 他 程序 员 又 会 在 这 之 上 继续 增添 复杂 性 ， 完 成 自己 的 
工作 。 程 序 员 对 化 解 复杂 性 考虑 得 越 少 ， 程 序 就 越 难 懂 。 

于 是 程序 变 得 无 比 复杂 ， 最 终 没 办 法 找 出 其 中 的 各 种 问题 。 喷 气 式 飞 


机 差不多 也 有 这 人 么 复杂 ， 但 它们 的 造价 是 几 百 万 甚至 几 十 亿美 元 ， 而 
且 仔 细 排 查 过 错误 。 大 多 数 软件 的 售 价 只 有 SO ~ 100 美元 。 价 钱 这 
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现在 我 们 已 经 知道 了 未 来 的 重要 性 ， 也 知道 了 关于 未 来 有 些 东 西 是 我 
们 不 知道 也 不 可 能 知道 的 ， 那 么 ， 有 什么 是 确定 的 呢 ? 


可 以 确定 的 一 点 是 ， 随 着 时 间 的 流逝 ， 软 件 所 处 的 环境 会 变化 。 没 有 
东西 可 以 永恒 不 变 。 也 束 是 说 ， 软 件 必 须 随 环境 变化 而 变化 ， 才 能 适 
应 所 处 的 环境 。 


于 是 ， 我 们 得 到 了 变化 定律 (Law of Change): 


程序 存在 的 时 间 越 久 ， 它 的 某 个 部 分 需要 变化 的 可 能 性 束 
越 商 。 
未 来 是 无 穷 无 尽 的 ， 所 以 可 以 百 分 百 地 肯定 ， 靶 终 ， 程 序 的 每 个 部 分 
部 必须 变化 。 在 未 来 5 分 钟 里 ， 可 能 没有 哪个 部 分 需要 变化 ;在 未 来 
10 天 里 ， 茶 个 小 部 分 可 能 需要 变化 ， 在 未 来 20 年 ， 即 便 不 是 全 部 都 
必须 变化 ， 很 可 能 大 部 分 也 必须 变化 。 


哪些 部 分 会 发 生变 化 很 难 预测 ， 为 什么 要 变化 也 很 难 预 油 。 很 可 能 你 
的 程序 是 为 4 EFFES, HARE NADIA 18 RRA RE, ih 
可 能 你 的 程序 是 写 给 高 中 生 的 ， 但 是 高 中 教育 的 质量 会 差 到 没有 学 生 
懂 你 的 程序 。 

关键 在 于 ， 你 并 不 需要 去 预测 什么 会 变化 ， 你 需要 知道 的 是 ， 变 化 必 
然 会 发 生 。 程 序 应 该 保证 尽 可 能 合理 的 灵活 性 ， 这 样 ， 不 管 未 来 发 生 
什么 变化 ， 都 可 以 应 付 得 了 。 


5.1 ”真实 世界 中 程序 的 变化 

来 看 一 份 关于 真正 的 程序 随时 间 流 逝 而 变化 的 数据 吧 。 某 个 程序 包含 
数 百 个 文件 ， 但 是 一 页 篇 幅 列 不 完 所 有 的 文件 ， 所 以 这 里 选取 了 4 个 
文件 作 样 本 。 表 5-1 列 出 了 它们 的 变化 详情 。 
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， 没 有 人 有 足够 的 时 间 和 资源 ， 在 几乎 无 限 复杂 的 系统 里 找到 所 
A ei gi. 


PL, ER" BEAT, WES LRT AA 
解 。 因 为 他 写 的 东西 都 很 好 懂 ， 所 以 要 找 出 bug 是 相当 容易 的 。 


这 个 关于 简单 性 的 想法 有 时 被 误解 为 ， 程 序 不 应 当 包含 太 多 代码 ， 或 
者 是 不 应 当 使 用 先进 技术 。 这 么 想 是 不 对 的 。 有 时 候 ， 大 量 的 代码 也 
可 以 带 来 简单 ， 只 不 过 增加 了 阅读 和 编写 的 工作 量 而 已 ， 这 是 完全 正 
常 的。 你 只 要 保证 ， 那 些 大 段 的 代码 提供 了 化 解 复杂 性 所 必须 的 简短 
注释 ， 就 足够 了 。 同 样 ， 通 常 来 说 ， 更 先进 的 技术 只 会 让 事情 更 简 
单 ， 只 是 一 开始 你 得 学 习 ， 所 以 整个 过 程 可 能 没 那么 简单 。 


有 些 人 相信 ， 把 程序 写 得 简单 所 花 的 时 间 ， 要 比 写 “能 用 就 好 ”的 程 
序 更 多 。 其 实 ， 花 更 多 的 时 间 把 程序 写 简单 ， 相 比 一 开始 随意 拼凑 些 
代码 再 花 大 量 的 时 间 去 理解 ， 要 快 得 多 。 这 个 问题 说 起 来 轻巧 ， 显 得 
轻描淡写 ， 其 实 软件 开发 的 历史 教训 很 多 ， 诸 多 事例 已 经 反复 证 明了 
这 一 点 。 许 多 大 型 程序 的 开发 之 所 以 会 停 少数 年 ， 就 是 因为 -开始 没 
有 做 好 ， 结 果 必 须 等 上 这 么 长 的 时 间 ， 才 能 给 之 前 开发 出 来 的 怪物 加 
上 新 功能 。 

正 因为 如 此 ， 计 算 机 经 常 出 问题 一 因为 大 多 数 程序 都 有 这 个 问题 ， 
许多 程序 员 在 写 程序 时 并 没有 化 解 复杂 性 。 是 的 ， 这 么 做 很 难 。 但 是 
如 果 程 序 员 做 不 到 这 一 点 ， 设 计 出 的 系统 过 于 复杂 、 经 常 出 问题 ， 用 
户 在 使 用 的 过 程 中 就 会 径 受 无 穷 无 尽 的 折磨 这 么 一 比 ， 把 程序 写 简 
WAR THA. 


12 程序 究竟 是 什么 ? 


大 多 数 人 说 的 “计算 机 程序 ， 其 实 有 完全 不 同 的 定义 


1) 给 计算 机 的 一 系列 指令 
(2) 计算 机 依据 指令 进行 的 操作 















































第 一 种 定义 是 程序 员 写 程序 时 所 用 的 。 弟 二 种 定义 是 使 用 程序 的 普通 
用 尸 所 用 的 。 程 序 员 命令 计算 机 ; 在 屏幕 上 显示 一 头 猪 。 这 就 是 第 一 
种 定义 ， 它 包含 者 干 指令 。 计 算 机 接收 到 指令 之 后 ,会 控制 电信 号 ， 
在 屏 医 上 显示 一 头 猪 。 这 是 后 一 种 定义 ， 即 计算 机 执行 的 操作 。 程 序 
员 和 用 万 都 会 说 自己 在 和 “计算 机 程序 ”打药 道 ， 但 是 他 们 的 用 法 
是 很 不 一 样 的 。 程 序 员 面 对 的 是 字母 和 符号 ， 用 户 看 到 的 是 最 终结 
果 一 一 计算 机 执行 的 操作 。 


所 以 ， 计 算 机 程序 其 实 是 这 两 者 的 混合 体 : 程序 员 的 指令 、 计 算 机 执 
A SE ES E E 
不 需要 执行 操作 ， 就 设 必要 去 写 代 码 了 。 这 就 好 像 在 生活 里 ， 你 列 了 
一 张 购物 单 (相当 于 指令 )， 告 诉 自己 该 买 哪些 东西 。 如 果 你 只 是 列 了 
单子 ,但 没 去 商店 ， 单 子 吏 没 有 任何 意义 。 指 令 必 须 得 到 实际 的 结果 ，。 


但 十 ， 到 购物 单 和 写 程序 有 显 看 的 区 别 。 如 来 购物 单列 得 很 乱 ， 只 不 
过 会 降低 买 东 西 的 速度 。 但 是 如 采 程 序 写 得 很 乱 ， 实 现 香 终 的 目标 就 
显得 尤其 困难 。 为 什么 昵 ? 因 为 购物 单 是 重 单 短小 的 ， 用 和 完 就 可 以 扎 
掉 。 而 程序 是 很 复 洒 很 庞大 的 ， 你 可 能 还 需要 维护 很 多 年 。 所 以 ， 同 
FEAT, GUERRAS EA MERA LA MR, ERA 
RICE Ae 


而 且 ， 除 软件 开发 之 外 ， 设 有 任何 领域 的 指令 和 结 条 联系 得 这 么 紧 
e EA, AMARGO.) AREA, AAA 
很 长 的 时 间 才 会 执行 。 比 如 设计 房子 ， 建 筑 师 首先 给 出 指令 一 一 也 就 
古 监 图 。 这 份 监 图 经 很 多 人 的 手 ， 过 了 很 长 的 时 间 ， 才 能 建 起 真正 的 
房子 。 所 以 ， 羽 子 是 大 家 所 有 人 解读 建筑 师 指 令 的 结果 。 相 反 ， 写 程 
序 时 ， 在 我 们 和 计算 机 之 则 没有 任何 人 人 。 我 们 下 计算 机 干什么 ， 就 会 
得 到 怎样 的 结果 ， 计 算 机 绝对 服从 命令 。 结 果 的 质量 完全 取决 于 机 器 
的 质量 、 我 们 想法 的 质量 ， 代 码 的 质量 ， 


在 这 三 个 因素 当中 ， 代 码 的 质量 是 如 今 软件 工程 需要 面 对 的 最 重要 问 
题 。 根 据 这 一 局 ， 以 及 上 面 所 到 的 其 他 原因 ， 本 书 的 大 部 分 和 内 容 部 在 
论述 如 何 提高 代 但 质量 。 会 有 一 些 地 方 入 及 机 缚 的 质量 和 权 法 的 质量 ， 
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虽然 花 了 这 么 多 的 时 间 来 谈 代 码 ,， 但 也 很 容易 忘记 ， 改 善 代码 质量 
的 一 切 努 力 都 是 为 了 获得 更 好 的 结果 。 本 书 绝 不 姑息 任何 差劲 的 结 
朵 一 一 我 们 学 习 提 高 代码 质量 的 全 部 原因 都 在 于 ， 要 想 改 进 结果 ， 提 
高 代码 质量 是 最 重要 的 问题 。 

高 代码 质量 的 科学 方法 。 


Hl, RKIISFEFEN, HER 

















事实 、 规 则 、 条 例 、 定 义 


MY Sx B 








事实 : 好 程序 员 和 差 程序 员 的 差别 就 在 于 理解 能 力 。 差 劲 的 程序 员 
不 理解 自己 做 的 事情 ， 优 秀 的 程序 员 则 相反 。 
- 条 例 ;“ 好 程序 员 ” 应 当 竟 尽 爹 力 ， 把 程序 写 得 让 其 他 程序 员 容 易 
理解 。 
dl EX: 程序 就 是 
(1) 给 计算 机 的 一 系列 指令 ， 
(2) 计算 机 依据 指令 进行 的 操作 ， 
。 EX: 软件 系统 中 任何 与 架构 有 关 的 技术 决策 ， 以 及 在 开发 系统 中 
所 做 的 技术 决策 ， 都 可 以 归 到 “软件 设计 ”的 范畴 里 。 
*。 事实 : 每 个 写 代码 的 人 都 是 设计 师 ，。 
© 事实 : 设计 与 民主 无 关 ， 它 应 当 由 个 人 完成 。 
© 事实 : 软件 设计 是 有 章 (规则 ) 可 循 的 ， 它 们 可 以 被 认识 ， 
理解 。 规 则 是 恒久 不 变 的 ， 是 基本 的 事实 ， 而 且 确 实 可 行 。 
© 规则 : 软件 的 目的 就 是 帮助 其 他 人 。 
© 事实 : 软件 设计 的 目的 如 下 。 
(1) 确保 软件 能 提供 尽 可 能 多 的 帮助 ， 
(2) 确保 软件 能 持续 提供 尽 可 能 多 的 帮助 ; 
(3) 设计 程序 员 能 尽 可 能 简单 地 开发 和 维护 的 软件 系统 ， 这 样 的 系统 
才能 为 用 户 提 供 尽 可 能 多 的 帮助 ， 而 且 能 持续 提供 尽 可 能 多 的 帮助 
Va + Ve 


E, + En 
这 是 软件 设计 的 主要 规则 ， 也 可 以 这 么 表达 :; 


改变 的 合意 程度 (可行 性 )， 正 比 于 软件 当前 价值 与 未 来 价值 之 和 ， 
有 反比 于 实现 成 本 和 维护 成 本 之 ; 


随 着 时 间 的 推移 ， 这 个 方程 式 会 简化 为 ; 


Vi 
D=-L 
En 


这 说 明 ， 降 低 维护 成 本 比 降低 开发 成 本 更 重要 。 
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。 条 例 : 要 做 多 少 设计 ， 应 当 正 比 于 未 来 软件 能 够 持续 为 人 们 提供 帮 
助 的 时 间 的 长 度 。 

。 条 例 : 未 来 的 某 些 事情 ， 是 我 们 所 不 知道 的 。 

。 事实 : 程序 员 犯 的 最 常见 也 是 最 严重 的 错误 ， 就 是 在 其 实 不 知道 未 
来 的 时 候 去 预测 未 来 。 

。 SP: 最 安全 的 情况 是 ， 完 全 不 尝试 预测 未 来 ， 所 有 有 的 设计 决策 都 
应 当 根 据 当 前 确切 知道 的 信息 来 做 。 

。 EM: 变化 定律 : 程序 存在 的 时 间 越 久 ， 它 的 某 个 部 分 需要 变化 的 
可 能 性 越 大 。 

。 事实 : 在 落实 变化 法 则 时 ， 软 件 设计 师 容易 犯 的 三 个 错误 (也 就 是 
本 书 中 的 “三 大 缺陷 ”) 是 : 
(1) 编写 不 必要 的 代码 
(2) 代码 难以 修改 
(3) 过 分 追求 通用 

。 条 例 ， 直到 真正 要 用 了 才 编 写 代码 ， 清 理 掉 用 不 到 的 代码 。 

。 条 例 ， 代码 的 设计 基础 ， 应 当 是 目前 所 知 的 信息 ， 而 不 是 你 认为 未 
来 要 发 生 的 情况 。 

。 事实 : 如果 设 计 让 事情 更 复杂 ， 而 不 是 更 人 简单， 就 犯 了 过 度 工程 的 
错误 。 

。 条 例 : 在 考虑 通用 时 ， 只 需要 考虑 当前 的 通用 需求 。 

。 条 例 : ARMAR AMAR, AULA ARA. 

© 条 例 : 缺陷 概率 法 则 ;在 程序 新 增 缺陷 的 可 能 性 与 代码 修改 量 成 正比 。 

。 条 例 : 最 好 的 设计 ， 就 是 能 适应 外 界 尽 可 能 多 的 变化 ， 而 软件 自身 
的 变化 要 尽 可 能 少 。 

。 条 例 : 永远 不 要 “修正 ”任何 东西 ， 除 非 它 真 的 有 问题 ， 而 且 有 证 
据 表 明 问 题 确实 存在 。 

。 条 例 : 理想 情况 下 ， 任 何 系统 里 的 任何 信息 ， 都 应 当 只 存在 一 次 。 

。 规则 : HEE: 软件 任何 一 部 分 的 维护 难度 ， 反 比 于 该 部 分 的 简 
洁 程 度 。 

。 事实 : 简洁 是 相对 的 。 

。 条 例 : 如 果 你 真 的 希望 成 功 ， 最 好 是 把 产品 简化 到 傻子 也 能 懂 。 
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。 条 例 ， 要 保持 一 致 。 


。 KA: 要 应 付 系统 中 的 复杂 性 ， 可 以 将 系统 分 


AN: EUER poi ini Bann 宇 白 排 布 。 
能 够 完整 表达 其 意义 或 手 述 其 功能 ， 但 不 





GEA, CLARE. 
条 例 ， 代 码 应 当 解释 程序 为 什么 这 么 做 ， 而 不 是 它 在 做 什么 ， 
条 例 : 简洁 离 不 开设 计 。 

”条 例 : 你 可 以 这 样 增加 复杂 性 








， 精 楼 的 设计 或 者 不 做 设计 ， 
a 重新 发 明 轮 子 ， 
" 背离 软件 原来 的 用 途 ， 


nial — 









mers tS AES AE SE 也 就 意味 着 深 藏 在 表面 
的 复杂 之 下 ， 设 计 出 了 问题 

条 例 : 在 复杂 性 面前 ， 问 问 自己 “真正 要 解决 的 问题 是 什么 "。 

。 Aal: 大 多 数 麻 烦 的 设计 问题 ， 都 可 以 用 在 纸 上 画 图 或 写 出 来 的 办 
法 找到 答案 。 








解 成 独立 的 小 部 分 ， 





逐步 重新 设计 。 

。 事实: 所 有 可 行 的 简化 ， 其 核心 问题 都 是， 怎么 做 ， 才 可 以 让 事情 
处 理 或 是 理解 起 来 更 容易 。 

+ 条 例 ， 如 果 遇 到 不 可 解决 的 复杂 性 ， 在 程序 外 面 妥 善 包装 上 一 层 ， 
让 其 他 程序 员 更 容易 使 用 和 理解 。 

AB: 推倒 重 来 只 有 在 一 些 非常 有 限 的 情况 下 才 是 可 以 接受 的 。 

。 规则 ， 测 试 定律 ， 你 对 软件 行为 的 了 解 程度 ， 等 于 你 真正 测试 它 的 
程度 。 

ABl: 除非 亲自 测试 过 ， 和 否则 你 不 知道 软件 是 否 能 正常 运行 。 
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1 ur 
"i sn ae on an rer KR Be a re Be =F ES Ej Re oF ds A a En ee h im = AS = = y kalt r= = aa E F EHE Be? sere = 55 aa I=, A EL BEE hr: 2 Se ; a a J See cae r ee re Er in E ei a u = 一 Meet a pa nes ie 5 mal ] A + 5 A E ae TAT BI E ep o ‘a Ea = 
ei se Pr ee A A A RER RE RUE SEEN ma are © hh he EL SE A ne 


一 一 最 前 治 的 IT 类 电子 书 发 售 平台 


电子 出 版 的 时 代 已 经 来 临 。 在 许多 出 版 界 同行 还 在 犹 殉 簿 律 的 时 候 ， 图 灵 社 区 已 经 采 
取 实 际 行动 拥抱 这 个 出 版 业 巨变 。 作 为 国内 第 一 家 发 售 电子 图 书 的 IT 类 出 版 商 ， 图 灵 社 区 “ 
目前 为 读者 提供 两 种 DRM-free 的 阅读 体验 : 在 线 阅 读 和 PDF。 

相 比 纸 质 书 ， 电 子 书 具有 许多 明显 的 优势 。 它 不 仅 发 布 快 ， 更 新 容易 ， 而 且 尽 可 能 采 © 
用 了 彩色 图 片 (即使 有 的 书 纸 质 版 是 黑白 印刷 的 ) 。 读 者 还 可 以 方便 地 进行 搜索 、 剪 贴 、 
复制 和 打印 。 

图 灵 社 区 进一步 把 传统 出 版 流程 与 电子 书 出 版 业务 紧密 结合 ， 目 前 已 实现 作 译 者 网 上 
交 稿 、 编 辑 网 上 审 稿 、 按 章 发 布 的 电子 出 版 模式 。 这 种 新 的 出 版 模式 ， 我 们 称 之 为 “敏捷 
出 版 ”， 它 可 以 让 读者 以 较 快 的 速度 了 解 到 国外 最 新 技术 图 书 的 内 容 ， 弥 补 以 往 翻 译 版 技 
术 书 “出 版 即 过 时 ”的 缺憾 。 同 时 ， 敏 捷 出 版 使 得 作 、 译 、 编 、 读 的 交流 更 为 方便 ， 可 以 
提前 消灭 书稿 中 的 错误 ， 最 大 程度 地 保证 图 书 出 版 的 质量 。 


优惠 提示 : 现在 购买 电子 书 ， 读 者 将 获 赠 书 款 20% 的 社区 银子 ， 可 用 于 兄 换 纸 质 样 书 。 


一 一 最 方便 的 开放 出 版 平台 


妈 灵 社区 向 读者 开放 在 线 写 作 功 能 ， 协 助 你 实现 目 出 版 和 开源 出 版 的 梦想 。 利 用 “ 合 
集 ” 功 能 ， 你 就 能 联合 二 三 好 友 共 同 创作 一 部 技术 参考 书 ， 以 免费 或 收费 的 形式 提供 给 读 
者 。( 收费 形式 须 经 过 图 灵 社 区 立项 评审 。 ) 这 极 大 地 降低 了 出 版 的 门 梓 。 只 要 你 有 写作 
的 意愿 ， 图 灵 社 区 就 能 帮助 你 实现 这 个 梦想 。 成 熟 的 书稿 ， 有 机 会 人 选 出 版 计划 ， 同 时 出 
版 纸 质 书 。 

图 灵 社 区 引进 出 版 的 外 文 图 书 ， 都 将 在 立项 后 马上 在 社区 公布 。 如 末 你 有 意 翻 译 哪 本 
图 书 ， 欢 迎 你 来 社区 申请 。 只 要 你 通过 试 译 的 考验 ， 即 可 签约 成 为 图 天 的 幸 者 。 当 然 ， 要 
想 成 功 地 完成 一 本 书 的 翻译 工作 ， 是 需要 有 坚强 的 角力 的 。 


一 一 最 直接 的 读者 交流 平台 

在 图 灵 社 区 ， 你 可 以 十 分 方便 地 写作 文章 、 提 交 勤 误 、 发 表 评 论 ， 以 各 种 方式 与 作 译 
者 、 编 辑 人 员 和 其 他 读者 进行 交流 互动 。 提 交 勘 误 还 能 够 获 赠 社区 银子 。 

你 可 以 积极 参与 社区 经 常 开展 的 访谈 、 乐 译 、 评 选 等 多 种 活动 ， 赢 取 积分 和 银子 ， 积 
累 个 人 声望 


5-1 ”随时 间 流 逝 的 文件 变化 
文件 1 文件 2 文件 3 = 
分 析 时 常 “5 年 2 个 月 8 年 13 年 3 个 月 13 年 4 个 月 
初始 行 数 。 423 227 309 

未 变化 行 数 271 
当前 行 数 644 
省 长 数 241 











修改 行 数 124 413 1382 3556 
总 变化 行 数 675 1709 3047 11107 
变化 率 1 .6X 8.9x 13x 36x 


表 中 各 项 指标 解释 如 下 。 


分 析 时 长 : 文件 存在 的 时 间 长 度 。 

初始 行 数 ， 文件 最 初 包含 多 少 行 。 
REMITA: 与 最 初 相 比 ， 没 有 变化 的 行 数 。 
当前 行 数 : 读 分 析 周 期 结束 时 ， 文 件 的 行 数 。 
变化 数 :“ 当 前 行 数 ”与 “最 初 行 数 ” 之 间 的 差额 。 

变化 次 数 : 程序 员 修 改 文件 的 次 数 (一 次 更 改 可 以 涉及 多 行 )。 通 常 
一 次 对 应 着 一 次 bug 修正 或 者 添加 一 项 新 功能 等 。 

新 增 行 数 : 总 的 来 看 ， 文 件 累计 新 增 了 多 少 行 ，。 

删除 行 数 : 总 的 来 看 ， 文 件 累计 删除 了 多 消 行 。 

修改 行 数 : 总 的 来 看 ， 文 件 累计 修改 了 多 少 行 。 

总 变化 行 数 : 新 增 行 数 、 删 除 行 数 、 修 改行 数 的 总 和 。 

变化 率 :“ 总 变化 行 数 ” 与 “最 初 行 数 ” 之 间 的 比率 。 

上 面 所 说 的 “ 行 数 ”"， 包 含 文件 的 每 一 行 : 代码 、 注 释 、 文 档 、 空 行 。 
如 果 忽 略 注释 、 文 档 、 空 行 ， 就 会 发 现 一 处 显著 差别 :“ 未 变化 行 数 ” 
会 比 其 他 几 个 指 慰 少 得 多 〈 也 就 是 说 ， 示 变化 的 行 儿子 都 是 注释 、 文 
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看 这 张 表 格 ， 应 当 弄 清楚 的 最 重要 一 点 是 ， 在 一 个 软件 项 目 中 ， 许 多 
变化 会 发 生 。 随 着 时 间 的 流 地 ， 代 码 的 每 一 行 都 需要 修改 的 概率 越 来 
MK; 然而 ， 你 并 不 能 精确 预测 要 修改 什么 ， 什 么 时 候 要 修改 ， 或 者 
要 修改 多 少 。 这 4 个 文件 的 变化 方式 各 不 相同 (只 看 数字 也 可 以 发 现 
这 点 ) ， 但 是 它们 都 发 生 了 相当 大 的 变化 。 


关于 这 些 指标 ， 还 有 一 些 有 趣 的 补充 。 


观察 变化 率 可 以 发 现 , 修改 文件 的 工作 量 比 最 初 写 文件 时 的 还 要 大 。 
当然 ， 行 数 这 个 指标 不 能 崔 确 反映 工作 量 ， 但 是 它 可 以 体现 一 般 趋 
势 。 有 时 候 变 化 率 很 高 ， 比 如 ， 文 件 4 的 总 变化 行 数 是 初始 行 数 的 
36 倍 。 

每 个 文件 中 的 未 变化 行 数 ， 比 起 “初始 行 数 ” 都 是 相当 小 的 ， 比 起 
“当前 行 数 ” 来 就 更 小 了 。 

文件 随时 间 逐 尖 增 长 只 是 一 方面 ， 还 有 很 多 其 他 变 人 化。 比如， 文件 
3 在 13 年 后 只 有 161 行 ， 但 是 总 变化 行 数 却 有 3047, 

总 变化 行 数 通 常 比 当前 行 数 要 大 。 也 就 是 说 ， 如 果 文 件 已 经 存在 了 
足够 长 的 时 间 ， 你 通常 会 去 做 修改 ， 而 不 是 新 增 代码 。 


© 观察 文件 3 的 指标 可 以 发 现 ， 修 改行 数 要 多 于 原始 行 数 与 新 增 行 数 之 


和 。 相 比 新 增 一 行 ， 修 改 已 有 东 行 的 频率 更 高 。 也 就 契 说 ， 文 件 中 有 
一 些 行 是 反复 修改 的 。 如 有 果 项 目 持续 的 时 间 很 长 ， 这 契 贡 见 的 情况 。 


以 上 列 出 的 只 是 部 分 的 信息 ， 关 于 这 些 数字 ， 还 可 以 做 其 他 十 多 有 趣 
的 分 析 。 我 们 或 励 你 去 探究 这 些 数 据 (或 者 是 分 析 目 己 项 目 里 的 类 似 
指标 ) 来 取得 进一步 的 收获 。 


a 


另 一 点 值得 学 习 的 有 趣 之 处 是 ， 回 顾 某 个 特定 文件 修改 历 
PE 
ww 


Re RRR TEE REA, MARA Ce ae 
个 文件 的 修改 历史 ， 请 回顾 整个 过 程 中 的 每 次 修改 。 问 问 
目 己 ， 最 初 写 这 个 文件 时 ， 你 能 预测 到 这 些 变化 吗 ， 是 否 
一 开始 写 好 就 能 够 减轻 后 期 的 工作 量 。 总 的 来 说 ， 就 十 要 
尝试 理解 每 次 修改 ， 看 看 是 否 能 从 中 得 到 一 些 关 于 软件 开 
发 的 新 的 收获 。 
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5.2 软件 设计 的 三 大 误区 
为 了 于 应 变化 定律 ， 软 件 设 计 师 常 第 会 掉 进 误区 。 其 中 有 3 个 误区 最 
AIN, ERRE AEREA: 


(1) 编写 不 必要 的 代码 
(2) 代码 难以 修改 
(3) 过 分 征求 通用 


5.2.1 编写 不 必要 的 代码 

如 仿 ， 软 件 设 计 中 有 一 条 第 见 的 规则 ， 则 做 “你 不 会 需要 它 ”(You 
Ain’t Gonna Need It)， 或 者 简称 为 YAGNI。 其 实 这 条 规则 的 意思 
是 ， 不 应 该 在 真正 的 需求 来 临 之 前 编写 那些 代码 。 这 条 规则 相当 不 
tar (EA SIR 了。 将 来 你 可 能 真 的 需要 这 些 人 代码, 但 是 既然 不 能 预 
测 未 来 ， 就 不 知道 现在 该 怎么 使 用 这 些 代 码 。 如 果 现 在 就 编写 这 些 代 
胎 ， 却 不 知道 该 怎么 用 ， 等 到 你 真 的 需要 使 用 时 ， 就 得 重新 设计 。 所 
以 应 当做 的 是 ， 省 下 重新 设计 的 时 间 ， 等 到 真正 需要 的 时 候 再 编写 那 
些 代 码 。 


在 需求 来 临 之 前 束 编 写 代 码 的 男 一 点 危险 是 ， 当 前 用 不 到 的 代码 很 可 
能 会 导致 “劣化 ” 。 因 为 代码 从 来 设 有 用 到 ， 它 很 可 能 与 系统 的 其 他 
BBS Me, FAME bug， 可 是 你 永远 也 不 知道 。 最 终 等 你 想 去 使 用 
的 时 候 ， 就 得 花 时 间 去 排 错 。 更 精 粒 的 是 ， 你 很 可 能 相信 这 些 从 来 设 
用 过 的 代码 是 正确 的 而 忽略 了 检查 ， 这 样 bug 就 留 给 了 用 户 。 其 实 ， 
这 条 规则 应 当 这 样 展开 : 


不 要 编写 不 有 是 必需 的 代码 ， 计 且 要 删除 没有 用 到 的 代码 。 


也 就 是 说 ， 你 还 需要 删除 所 有 用 不 到 的 代码 ， 如 果真 的 需要 ， 你 随时 
可 以 恢复 回来 。 











注 1: bit rot， 这 和 是 一 个 技术 术语 ， 意 指 随时 间 推 移 ， 软 件 劣 化 或 者 丢失 数据 的 现象 。 
— HET 


46 | 第 5 章 


出 于 一 些 其 他 原因 ， 还 是 有 人 会 认为 他 们 应 当 现在 就 编写 一 些 今后 才 
会 需要 的 代码 ， 或 者 古 保 留 暂 时 用 不 到 的 代码 。 有 些 人 相信 目 己 可 以 
抗拒 变化 定律 ， 只 要 某 个 功能 现在 有 任何 一 名 用 户 可 能 用 到 ， 就 为 此 
编写 代码 。 这 样 一 来 ， 程 序 在 未 来 就 不 需要 改变 或 改进 了 。 但 是 ， 这 
桩 做 古 不 对 的 。 只 要 一 直 有 人 用 ， 就 没有 什么 系统 可 以 永远 你 持 不 变 。 


还 有 人 认为 ， 现 在 多 做 些 工 作 ， 将 来 就 可 以 市 省 时 间 。 有 的 场合 下 这 种 
做 法 行 得 通 ， 但 这 不 包括 写 没 人 用 的 代码 的 情况 。 即 便 这 些 代 码 将 来 可 
以 派 上 用 场 ， 多 半 也 得 要 化 时 间 重 新 设计 ， 所 以 最 终 还 契 浪 费 了 时 间 。 


编写 不 必要 的 代码 : 真实 的 例子 


从 阐 ， 有 位 开发 人 员 就 叫 他 Max 吧 一 一 以 为 自己 可 以 不 受 这 条 规 
则 的 束 弹 。 他 的 程序 里 有 一 项 下 拉 菜 单 ， 用 户 可 以 从 中 选取 一 个 值 。 
使 用 该 软件 的 每 家 公司 部 会 自己 定制 这 个 菜单 的 选项 。 有 些 公司 定制 
的 选项 可 能 是 颜色 的 名 字 ， 还 有 些 可 能 是 城市 的 名 字 ， 当 然 还 有 其 他 
选项 。 所 以 ， 可 供 选择 的 项 应 当 存 储 在 每 家 公司 部 可 以 修改 的 地 方 。 


显然 存储 可 供 选 择 的 值 就 够 了 。 毕 竞 , 需要 做 的 就 是 这 些 。 但 是 
Max 决 定 多 走 一 步 : 既 要 保存 每 个 选项 值 ， 又 要 记录 每 个 值 当 前 是 
否 可 用 一 一 也 就 是 说 ， 用 户 现 在 能 否 选择 这 些 选项 ， 这 个 值 是 
否 临时 茶 用 了 。 


不 过 ，Max 一 直 没 写 任何 判断 选项 是 否 “ 可 用 的 代码 。 目 始 至 终 ， 
无 论 存 储 了 什么 数据 ， 所 有 的 选择 都 是 可 用 的 。 虽 然 Max 以 为 他 马上 
就 要 写 那 些 判 断 可 用 的 代码 了 ， 没准 儿 就 在 第 二 天 动手 。 


几 年 过 去 了 ,判断 “ 可 用 ”数据 的 代码 一 直 没 有 出 现 。 相 反 ， 数 据 
只 是 存在 那里 ， 没 有 任何 地 方 用 到 ， 却 会 干扰 其 他 人 的 认 知 ， 产 生 
错误 。 许 多 客户 和 开发 人 员 写 信 给 Max， 想 询问 为 什么 他 们 把 若干 
选项 设 定 为 “不 可 用 ”状态 时 ， 没 有 任何 变化 。 有 一 名 开发 人 员 锚 
误 地 假设 “可 用 ”字段 是 有 用 的 ， 甚 至 写 了 些 代 码 来 使 用 它 ， 即 便 
系统 的 其 他 部 分 都 没有 用 到 。 修 改 后 的 程序 到 了 客户 手 里 ， 客 户 就 
开始 报告 奇怪 的 问题 ， 需 要 花 大 力气 来 追踪 。 
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最 后 ， 有 些 开 发 人 员 要 动手 了 ， 他 们 告诉 自己 : 今天 就 要 实现 禁用 
选项 的 功能 。 但 是 他 们 发 现 ， 可 用 ”字段 的 设计 并 不 能 满足 自己 
的 需求 ， 所 以 他 得 花 相 当 多 的 时 间 重 新 设计 才能 实现 这 个 功能 。 


最 终 的 结果 是 : 若干 bug， 严 重 的 混 消 视听 和 困扰 ， 真 正 需 要 这 上 段 
代码 的 开发 人 员 还 得 做 大 量 的 工作 。 这 还 不 算是 严重 违背 规律 的 行 
为 | 倘若 严重 违背 这 条 规律 ， 会 导致 更 加 恶劣 的 后 果 ， 包 括 项 目 延 
期 ， 造 成 巨大 的 灾难 ， 整 个 项 目 黄 至 会 因此 完蛋 。 





5.2.2 ”代码 难以 修改 


软件 项 目的 一 大 杀手 就 是 所 谓 的 “僵化 设计 (rigid design). itz 
说 ， 程 序 员 写 出 来 的 代码 很 难 修改 。 僵 化 设计 有 两 大 原因 : 


(1) 对 未 来 做 太 多 假设 
(2) 不 仔细 设计 就 编写 代码 


举例 ， 太 多 关于 未 来 的 假设 
一 家 机 构 一 一 就 叫 “ 老 兵 医院 ” 吧 一 一 布 望 开 发 一 个 程序 。 他 们 巴 
它 “ 医 疗 保健 系统 ”。 在 设计 该 系统 之 前 ， 医 院 决 定编 制 文 档 详细 
说 明 整 个 系统 的 实现 细节 ， 他 们 花 了 一 年 来 写 这 份 文档 ， 确 中 系统 
中 的 每 一 点 决策 。 


然后 ， 开 发 人 员 依 照 文档 ， 花 了 3 年 把 系统 开发 出 来 。 在 开发 时 ， 
他 们 发 现 文档 中 的 设计 有 地 方 自 相 了 矛盾 ， 有 地 方 不 完整 ， 还 有 地 方 
很 难 实 现 。 可 是 这 份 文 档 是 医院 花 了 一 整 年 编写 的 ， 所 以 开发 人 员 
不 能 再 等 一 年 让 医院 来 修订 。 于 是 ， 开 发 人 员 还 是 尽 可 能 按照 文档 
完成 了 整套 系统 。 

系统 完成 之 后 就 进行 了 首次 交付 。 但 是 现在 ， 医 院 的 情形 已 经 大 不 
相同 了 。 距 离 最 初 的 设计 已 经 过 去 了 四 年 ， 用 户 真 正 使 用 这 套 系 统 
时 ， 发 现 自己 需 要 的 已 经 完全 不 同 了 。 然 而 系统 是 由 几 十 万 行 代码 
构成 的 ， 其 设计 也 是 严格 按照 文档 来 构建 的 ， 除 非 再 花 上 几 个 月 其 
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| 至 几 年 ， 否 则 系统 很 难 改变 。 
所 以 ， 公 司 开始 为 新 系统 重 写 设计 文档 ， 把 之 前 的 过 程 重新 来 过 。 


医院 的 错误 就 在 于 他 们 试图 预测 未 来 。 他 们 假设 文档 中 所 作 的 每 一 
点 改动 都 是 对 最 终 用 户 真 正 有 用 的 ， 而 且 到 系统 完成 的 时 候 还 会 
一 直 有 用 。 未 来 真 的 到 来 时 ， 却 完全 与 之 前 预测 的 不 一 样 ， 结 果 几 
百 万 美元 打 了 水 漂 。 | 
更 好 的 办 法 是 每 次 只 确定 一 个 或 者 少数 几 个 需求 ， 然 后 立刻 让 开发 人 
员 实现 它 。 在 开发 过 程 中 ， 用 户 可 以 扮演 开发 人 员 的 角色 ， 反 复 进 
行 沟通 。 上 次 确定 的 功能 实现 并 发 布 之 后 ， 就 可 以 继续 处 理 其 他 的 功 | 
能 。 这 样 ， 最 终 得 到 的 系统 是 设计 良好 的 ， 完 全 满足 用 户 需求 的 。 





举例 : 缺乏 设计 的 编码 


| 某 开 发 人 员 被 要 求 瑟 一 个 程序 ， 供 人 们 记录 他 们 完成 任务 的 进度 。 
为 了 在 系统 里 新 建 一 个 任务 ， 用 户 需 要 在 表单 里 填写 某 些 信息 ， 
比如 任务 的 概要 信息 、 现 在 的 进度 。 这 些 信息 存储 在 数据 库 里 。 然 

后 ， 随 着 时 间 的 推移 ， 大 家 一 点 点 地 记录 任务 的 进展 ， 最 终 标 汪 自 
己 完 成 了 整个 任务 。 
软件 里 有 小 叫 Status 的 字段 ， 用 杂记 录 当前 任务 的 进度 。 这 个 字段 的 
值 有 No Work Done (AK) ，In Progress (进行 中 ) , On Hold (4% 
m). Complete (完成 ) 。 如 果 值 为 No Work Done， 只 能 改 为 In 

Pn n Progress， 只 能 改 为 On Hold 或 者 Complete。 如 
果 值 为 Complete， 只 可 以 改 回 为 In Progress, | 
程序 里 还 有 其 他 10 个 字段 需要 用 到 类 似 的 规则 ， 每 个 都 包含 任务 某 
75958 (KB, MEER). 

为 实现 这 些 规则 ， 程 序 员 专门 写 了 一 段 长 长 的 代码 ， 筷 没有 任何 
结构 ， 而 是 独立 保存 为 一 个 文件 。 这 段 代 码 验 证 每 个 字段 的 正确 

性 ， 也 仅 为 完成 验证 功能 而 设计 。 比 如 ， 每 次 检查 状态 是 否 为 
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Complete， 就 在 代码 里 硬 编码 写 上 Complete。 同 样 ， 编 写 代 码 时 也 
不 考虑 重用 。 允 到 相似 的 字段 ， 开 发 人 员 就 复制 -粘贴 代码 ， 然 后 
有 针对 性 地 做 些小 修改 。 


这 段 代 码 当 然 能 用 。 文 件 有 3000 行 ， 但 基本 与 设计 绝缘 。 
几 修 月 过 去 了 ， 开 发 人 员 离 开 了 项 目 。 


之 后 ， 一 名 新 开发 人 员 被 安排 来 维护 这 个 项 目 。 他 很 快 发 现 这 段 代 
码 很 难 修改 ， 如 果 修改 其 中 的 一 部 分 ， 就 得 对 其 他 很 多 地 方 做 同样 
的 修改 ,才能 保证 程序 正常 运行 。 更 糟 粮 的 是 ， 这 些 类 似 的 部 分 分 
得 很 散 ， 既 没有 注释 ， 也 没有 逻辑 一 一 每 次 需要 修改 时 ， 除 了 阅读 
整个 文件 外 别 无 他 法 。 

然后 ， 客 户 要 求 添加 新 的 功能 。 一 开始 ， 新 来 的 这 名 开发 人 员 尽 自 
已 努力 来 完成 新 功能 ， 他 在 文件 中 增加 了 更 多 代码 。 代 码 行 数 达到 
了 5000 行 。 


最 终 ， 客 户 要 求 的 某 些 功能 ， 原 有 设计 方案 已 经 无 法 实现 了 。 客 户 
希望 用 电子 邮件 更 新 任务 的 进度 ， 但 是 原 有 代码 只 能 处 理 表 单 。 原 
有 的 设计 是 紧密 地 与 表单 绑 定 的 一 一 它 无 法 对 接 电子 邮 件 。 

这 时 候 ， 竞 争 对 手 出 现 了 ， 他 们 的 任务 可 以 用 电子 邮件 来 更 新 。 结 
果 ， 原 有 的 软件 开始 流失 客户 。 


这 个 项 目 之 所 以 能 维持 下 去 ， 完 全 是 因为 有 两 名 开发 人 员 花 了 一 整 年 
重新 设计 了 这 个 文件 ， 降 低 了 它 的 修改 难度 。 在 重新 设计 时 ， 他 们 尽 
自己 努力 来 满足 其 他 新 需求 ， 但 大 多 数 时 间 还 是 花 在 重新 设计 上 。: 





要 避免 伪 化 设计 ， 就 应 当做 到 : 


设计 程序 时 ,应当 根 据 你 现在 确切 知道 的 需求 ， 而 不 是 你 
认为 未 来 会 出 现 的 需求 。 


注 2: Bugzilla 中 的 这 个 文件 叫 process_bug.cgi。 这 个 故事 做 了 删 略 ， 但 是 数字 (也 就 是 
代码 的 行 数 ， 以 及 用 来 修复 的 时 间 ) 却 是 大 致 准确 的 。 如 果 你 想 看 关于 重新 设计 整 
个 项 目的 完整 故事 ， 看 看 这 一 切 是 怎么 发 生 的 ， 可 以 参考 这 个 链接 : https://bugzilla. 
mozilla.org/showdependencytree.cgi?id=367914&hide_resolved=0, 
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基于 当前 确 知 的 需求 来 设计 ， 同 时 兼顾 未 来 需求 的 可 能 性 。 如 果 你 确 
切 地 知道 ， 系 统 需 要 完成 XX， 而且 只 是 义 ， 那 么 现在 设计 时 就 应 该 内 
考虑 完成 六。 你 应 该 记 住 ， 未 来 系统 可 能 还 要 提供 义 之 外 的 功能 ， 但 
是 现在 ， 系 统 只 应 该 完成 成。 


如 果 照 这 样 来 设计 ， 另 外 应 该 做 到 的 一 点 是 ， 保 持 每 次 改变 的 幅度 都 
很 小 。 如 果 要 做 的 改变 很 小 ， 设 计 的 难度 也 会 很 小 。 


这 不 是 说 不 要 做 规划 。 在 软件 设计 中 ， 一 定 程度 的 规划 征 非 基 有 价值 
的 。 但 是 ， 即 便 不 做 详细 的 规划 ， 只 要 你 能 保持 改变 的 幅度 很 小 ， 代 
码 也 很 容易 适应 不 确定 的 未 来 ， 就 没有 大 的 风险 。 


5.2.3 ”过 分 追求 通用 

既然 代码 将 来 要 修改 是 一 个 事实 ， 有 些 开 发 人 员 给 出 的 应 对 方案 就 
是 : 做 一 个 足够 通用 的 办 法 ， 保 证 (他 们 目 己 相信 ) 可 以 适应 未 来 任 
何 可 能 的 形势 。 我 们 称 这 种 做 法 为 “过 度 工程 (overengineering), 


按照 字典 的 定义 ，overengineering 就 是 over (意思 是 “过 分 了 ") 加 
engineer (意思 是 “设计 和 构造 )}。 根 据 这 种 解释 ， 过 床 工 程 意 思 就 
是 ,在 设计 或 者 构造 上 花 了 过 多 的 精力 ， 


等 一 下 ， 设 计 和 构建 过 多 了 ? 什么 是 “过 多 ”? 设计 不 是 一 件 好 事 




































是 的 ， 大 多 数 工程 都 应 该 在 设计 上 花 更 多 的 精力 ， 在 上 一 玉 中 “ 举 
Al: 缺乏 设计 的 编码 ”中 已 经 看 到 这 点 。 但 是 总 会 有 人 真 的 掉 到 过 度 
LENA EA TERN, MANE ALE 
轨道 沿 光 器 是 有 相当 挑战 的 工程 ， 它 耗费 不 非 ， 制 造 时 间 相 当 长 ， 
护 起 来 也 是 一 场 松 梦 。 你 能 想象 它 出 辐 题 的 时 候 应 当 垮 么 修复 吗 ? 


过 度 工程 还 有 其 他 几 个 辐 题 。 


(1) 因为 你 不 能 预测 未 来 ， 所 以 无 论 你 做 得 多 么 通用 ， 其 实 都 不 够 注 
足 未 来 要 面 对 的 真实 需求 。 
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(2) 如 果 你 的 代码 很 通用 ， 那 么 它 通常 不 能 从 用 户 的 角度 很 好 地 满足 规 
格 /需求 。 举 例 来 说 ,假设 你 设计 的 代码 把 所 有 输入 都 当成 二 进 制 字 
市 来 对 待 。 有 了 时候 这 代码 要 处 理 文本 ， 有 时候 要 处 理 图 厂 ， 但 十 它 只 
知道 输入 的 是 字 节 。 某 种 方面 来 说 ， 这 种 设计 不 错 : BER, 
自治 、 短 小 。 然 后 你 才 知 道 ， 代 码 根本 无 法 区 分 图 片 和 文本 一 一 这 就 
是 过 分 通用 了 。 用 户 传 进来 一 张 损坏 的 图 片 ， 得 到 的 错误 却 是 “传人 
的 字 节 错误 。 本 来 它 应 该 报告 “图 片 已 经 损坏 ， 但 是 你 的 代码 太 过 
通用 ， 无 法 给 出 这 种 提示 ( 太 过 通用 的 代码 很 多 时 候 无 法 满足 具体 的 
需求 ， 这 只 是 一 个 例子 )。 


(3) 太 过 通用 就 必须 写 很 多 不 需要 的 代码 ， 这 样 又 回 到 了 第 一 条 规律 。 


总 的 来 说 ， 如 果 你 的 设计 让 事情 更 复杂 而 不 是 变 侧 单 ， 就 走 在 做 过 度 
工程 。 如 果 你 只 需要 清理 蚁 穴 ， 动 用 轨道 激光 就 会 把 事情 大 大 搞 复 
杂 ， 其 实 只 用 一 点 蚂蚁 药 就 可 以 解决 问题 (假设 它 有 效 )。 


在 追求 通用 时 ， 应 当选 择 正确 的 事情 ， 选 择 正 确 的 方法 ， 这 定 成 功 的 
软件 设计 的 基础 。 然 而 ， 太 过 通用 ， 会 带 来 说 不 完 的 复杂 和 混乱 ， 也 
会 大 大 抬 高 维护 成 本 。 避 免 此 误区 的 办 法 ， 和 如 免 僵 化 设计 的 一 样 : 


仅仅 根据 目前 确 知 的 需求 来 考虑 通用 。 





250): 太 过 通用 
某 程 序 的 某 个 部 分 的 功能 是 ， 用 户 填写 一 张 表单 ， 程 序 就 发 送 几 百 
封 邮 件 。 这 个 部 分 的 程序 运行 起 来 很 慢 。 用 户 提交 表单 之 后 ， 程 序 
会 在 这 个 环节 停留 很 长 时 间 ， 等 待 发 送 完 所 有 的 邮件 。 
为 了 提高 这 个 环节 的 速度 ， 开 发 人 员 决 定 不 要 立即 发 送 所 有 的 邮 
件 ， 而 是 改 为 在 后 台 发 送 。 用 户 提 交 了 表单 之 后 ， 程 序 会 使 用 之 前 
存在 的 “Email Sender ”代码 来 发 送 邮 件 。 


负责 这 个 任务 的 开发 人 员 认 为 有 些 公 司 并 不 会 使 用 Email Sender, 
他 写 了 几 百 行 代码 ， 这 样 用 户 就 可 以 把 其 他 系统 以 插件 的 方式 塞 进 
来 ， 在 后 台 执 行 这 个 任务 。 从 没有 客户 提 过 这 个 需求 ， 开 发 人 员 只 
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是 猜测 未 来 会 有 人 要 求 这 种 灵活 性 。 

最 终 ， 首 席 架 构 师 叫 停 了 这 种 做 法 。 他 去 掉 了 所 有 与 “插件 AX 
的 代码 ， 因 为 根本 没有 证 据 表 明 有 人 会 那么 用 。 也 就 是 说 ， 没 有 任 
何 证 据 表明 目前 代码 需要 做 到 那么 通用 。 这 些 代码 删 卸 之 后 ， 修 改 
变 得 容易 多 了 ，。 

这 一 改变 已 经 过 去 了 四 年 ， 期 间 没 有 任何 客户 要 求 给 系统 做 插件 。 
也 就 是 说 ， 系 统 其 实 根 本 没有 必要 做 得 那么 通用 。 


5.3 ”渐进 式 开发 及 设计 
有 个 办 法 可 从 根本 上 训 免 这 三 大 误区 ， 这 就 是 “次 进 式 开 发 和 设计 。 
尼 要 求 按 照 尾 定 顺 序 ， 一 点 一 点 地 设计 和 构建 系统 。 


这 个 办 法 很 容易 举例 来 说 明 。 假 如 需要 开发 可 以 计算 加 减 乘除 的 计算 
ar BEF 


(1) 设计 一 个 只 能 进行 加 法 运算 的 系统 。 

(2) 实现 它 。 

(3) 修改 现 有 设计 ， 很 容易 就 可 以 支持 减法 运算 。 

(4) 实现 减法 运算 。 现 在 ， 系 统 只 能 进行 加 、 减 运算 。 

(5) 再 次 修改 现 有 系统 的 设计 ， 很 容易 就 可 以 支持 乘法 运算 。 

(6) 实现 乘法 运算 功能 。 现 在 的 系统 可 以 支持 加 、 减 、 乘 运算 了 。 

(7) 再 次 修改 系统 的 设计 ， 添 加 除法 运算 (现在 做 这 一 步 应 该 不 需要 
化 多 少 功 夫 『， 因 为 我 们 早 就 改进 了 设计 ， 添加 过 减法 、 乘 法 )。 

(8) 实现 除法 功能 。 这 时 候 得 到 的 就 是 一 开始 期 望 构建 的 完整 系统 ， 
而 且 拥 有 满足 需求 的 民 好 设计 。 


相 比 开始 就 建立 完整 的 系统 ， 一 次 性 构建 出 来 ， 这 种 开发 方法 需要 时 
间 更 少 ， 也 不 用 考 谍 过 多 。 如 未 你 习惯 其 他 的 开发 方法 ， 头 一 次 实践 
可 能 并 不 那么 容易 ， 但 是 经 过 锻炼 ， 用 起 来 就 会 变 容 易 。 


这 个 方法 的 精妙 之 处 赋 在 于 ， 它 征 根据 实现 的 顺序 来 决策 的 。 总 的 来 
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说 ， 在 其 中 的 每 个 阶段 ， 下 一 步 都 只 做 最 容易 的 事情 。 首 先 选 择 加 
法 ， 因 为 这 契 了 最 向 单 的 运算 1 其 次 选择 减法 ， 因 为 从 逻辑 上 说 ， 它 与 
加 法 只 有 很 小 的 差异 。 第 二 步 也 可 以 选择 乘法 ， 因 为 乘法 无 非 是 把 加 
法 执行 很 多 次 而 已 。 这 时 唯一 不 应 当选 择 的 就 是 除法 ， 因 为 从 加 法 到 
除法 的 步子 距离 太 远 了， 步子 太 大 了 。 而 且 ， 最 后 一 步 从 乘法 到 除法 
慷 非 第 向 单 的 ， 所 以 这 是 一 个 好 的 选择 。 


有 些 时 候 ， 你 甚至 需要 把 茶 个 单独 的 功能 拆 分 为 一 系列 小 的 、 人 向 单 的 
wR, aA ALAR (Ee EBL. 


显然 ， 这 里 混合 了 两 种 做 法 : — FU EFA RR, AH BPA 
痢 进 设计 。 痢 进 开发 是 一 种 通过 小 步骤 构建 整个 系统 的 办 法 。 在 上 
面 的 步骤 里 ， 实现 ”开头 的 每 一 步 都 是 源 进 开发 过 程 的 一 部 分 。 源 
进 设计 契 一 种 类 似 的 方法 ， 它 也 通过 一 系列 小 步骤 用 来 创建 和 改进 系 
统 的 区 计 。 在 上 面 的 步骤 里 ， 以 “修改 或 设计 开头 的 每 一 步 ， 
部 征 辣 进 设 计 过 程 的 一 部 分 。 


汤 进 开发 和 设计 并 不 古 唯 一 有 效 的 软件 开发 方法 但 古 它 无 疑 可 以 避 
免 之 前 列 出 的 三 大 误区 。 


54 | 第 5 章 





很 不 幸 ， 没 有 哪个 程序 员 可 以 不 犯错 误 。 不 错 的 程序 员 大 概 每 写 100 
行 代码 就 会 犯 一 个 错误 ， 最 优秀 的 程序 员 ， 在 状态 最 好 的 时 候 ， 每 写 
1000 行 代码 也 会 犯 一 个 错误 。 


也 就 是 说 ， 无 论 程序 员 的 水 平 是 高 还 是 低 ， 有 一 条 是 不 变 的 ， 写 的 代 
友 越 多 ， 引 入 的 缺陷 就 越 多 。 这 样 ， 就 可 以 得 出 “缺陷 概率 定律 ” 


在 程序 中 新 增 负 了 的 可 能 性 与 代码 修改 重 成 正比 。 
一 规则 之 所 以 重要 ， 是 因为 错误 妨碍 了 我 们 帮助 他 人 的 目标 ， 故 而 
应 当 避 免 。 而 且 ， 修 复 缺陷 也 是 维护 工作 的 一 种 。 所 以 ， 新 增 缺 陷 还 
会 抬 高 维护 成 本 


既然 存在 这 条 规则 ， 又 无 法 预测 未 来 ， 我 们 很 快 就 会 发 现 ， 相 比 大 的 
变化 ， 小 的 变化 维护 成 本 更 低 。 小 的 变化 = 更 少 的 缺陷 = 更 少 的 维护 。 


有 时候 ， 该 规则 也 会 被 非 正 式 地 表述 为 :“ 如 果 不 新 加 代码 ， 也 不 修 
改 代码 ， 就 不 会 产生 新 缺陷 。 


这 条 规则 的 有 趣 之 处 在 于 ， 它 似乎 与 “变化 定律 ” 相 矛 盾 一 软件 必 
须要 变化 ， 但 是 变化 又 会 引入 缺陷 。 这 种 矛盾 确实 存 企 ， 如 何在 两 者 
间 取得 平衡 ， 取 决 于 软件 设计 师 的 聪明 才智 。 实 际 上 ， 这 种 矛盾 恰恰 
说 明了 为 什么 需要 设计 ， 而 且 告诉 我 们 ， 理 想 的 设计 是 怎样 的 : 
最 好 的 设计 ， 就 是 能 过 应 外 界 尽 可 能 多 的 变化 ， 而 软件 自 
身 的 变化 要 尽 可 能 少 。 
这 个 说 法 ， 简 练 融 合 了 如 今 关于 优秀 的 软件 设计 的 各 项 知识 ， 


6.1 如果 这 不 是 问题 …… 


没 错 ， 如 果 不 添加 代码 ， 也 不 修改 代码 ， 就 不 会 引入 新 的 缺陷 ， 这 是 
软件 设计 中 的 一 条 主要 规律 。 不 过 ， 还 有 下 面 这 条 非常 重要 的 规律 与 
之 相关 ， 证 多 软件 设计 师 都 昕 过 芭 种 形式 的 表述 ， 虽 然 他 们 有 了 时候 会 
忘记 ; 
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永远 不 要 “修正 ”任何 东西 ， 除 非 它 真 的 有 问题 ， 而 且 有 
证 据 表明 问题 确实 存在 。 
在 动手 修正 问题 之 前 ， 获 得 证 据 是 很 重要 的 。 否 则 ， 你 的 辛 : 


可 能 解决 不 了 任何 人 的 问题 ， 或 者 你 很 可 能 会 ， 
的 代码 。 


如 来 没有 获得 证 据 就 动手 修正 问题 ， 很 可 能 只 会 添乱 。 修 改 现 有 系 
统 ， 可 能 会 霹 成 新 的 馈 误 。 而 且 ， 这 么 做 还 会 浪费 时 间 ， 增 加 程序 的 


BA, Ze 证 据 Ye? RRARAAP Rig: 一 点 击 红色 按钮 ， 
程序 就 贿 秆 了 。 好 ， 这 就 是 足够 充分 的 证 据 了 ! 否则 ， 你 可 能 需要 
CARETO, BARES mit. 


但 是 ， 如 果 只 有 一 个 用 户 报告 错误 ， 并 不 能 说 明 这 就 是 问题 。 有 时 候 
用 户 不 知道 你 的 程序 已 经 提供 了 某 些 功能 ， 所 以 希望 你 去 重复 实现 它 
们 。 举 例 来 说 ， 你 的 程序 可 以 按照 字母 顺序 来 排序 pp 
户 希 望 你 添加 一 个 功能 ， 





后 = ae je 























只 是 用 户 没 弄 清 楚 到 底 要 干什么 。 在 这 个 例子 里 ， HARPUARA 
单词 排序 的 功能 ， 他 可 能 认为 没有 实现 字母 排序 是 个 问题 。 用 户 还 可 
能 给 出 “证 据 ， 说 明日 己 无 法 排序 一 系列 字母 。 其 实 ， 问 题 仅仅 在 
于 ， 他 没有 意识 到 目 己 应 该 使 用 单词 排序 的 功能 ， 








Ha 





RPA RSA AIA, RUA PR EEE 
。 里 找到 自己 想 要 的 功能 。 这 才 是 真正 要 改进 的 地 方 。 





有 时候 用 户 会 报告 某 个 bug， 但 程序 其 实 是 完全 按照 预期 来 运行 的 。 
果真 如 此 ， 就 应 当 少数 服从 多 数 。 如 果 相 当 多 的 用 户 认为 某 个 行为 是 
bug, EXE bug， 如 果 只 是 少数 用 户 〈 比 如 一 两 个 ) 认为 它 是 bug， 
那么 它 就 不 算 bug. 


在 这 类 问题 上 ， 晶 有 名 的 错误 就 是 所 谓 的 “提前 优化 。 也 就 是 说 ， 
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有 些 开发 人 员 想 让 速度 尽 可 能 快 ， 所 以 ， 他 们 还 没 弄 清楚 速度 到 底 慢 
不 慢 ， 就 化 时 间 来 优化 程序 。 这 就 好 像 做 慈善 事业 时 ， 一 边 把 食物 送 
给 主人 ， 一 边 说 “我 们 只 是 斋 望 帮助 他 人 。 这 不 合 逻 辑 ， 对 吧 ? A 
为 这 样 征 在 解决 根本 不 存在 的 问题 。 


在 你 的 程序 中 ， 真 正 需 要 关注 速度 的 部 分 ， 应 该 局 限于 你 可 以 证 明 
的 、 真 正 让 用 户 体 会 到 有 性 能 问题 的 那些 部 分 。 对 程序 的 其 他 部 分 ， 
最 主要 关心 的 还 是 灵活 和 简洁 ， 而 不 是 速度 。 


要 违背 这 条 规律 有 成 千 上 万 种 办 法 ， 不 过 遵守 它 的 办 法 很 画 单 : 在 动 
手 解决 之 前 ， 真 正 拿 到 证 据 ， 证 明 问 题 确实 存在 。 


6.2 ”避免 重复 


在 软件 设计 中 ， 这 或 许 是 最 著名 的 条 例 。 其 他 资料 也 曾 提 过 该 条 例 ， 
侈 于 它 的 重要 性 ， 我 们 在 这 里 重申 : 
理想 情况 下 ， 任 何 系统 里 的 任何 信息 ， 都 应 当 只 存在 

一 次 ， 
假设 有 个 Password 字段 ， 在 你 的 程序 里 会 出 现在 100 个 用 户 界 面 上 。 
Qn Fe On ar BE EMMA Passcode 该 怎么 办 ? 如 果 字 段 名 统一 存储 在 程 
序 某 个 地 方 ， 就 只 需要 修改 一 行 代 码 。 但 如 果 每 次 都 是 硬 编码 写 上 
Password， 就 得 修改 100 次 。 
这 个 追 理 对 代码 段 也 同样 适用 。 我 们 不 应 该 复制 粘贴 代码 段 ， 相 反 ， 
应 该 使 用 各 种 编程 技巧 来 处 理 ， 让 各 处 的 代码 可 以 “使 用 ”(use)、 
“调用 ”(call)、“ 包 含 ”(include) 已 有 的 其 他 代码 。 
遵守 这 条 规则 的 一 个 强 有 力 的 理由 ， 就 是 缺陷 概率 定律 。 如 果 新 增 功 
能 时 可 以 重用 代码 ， 就 不 需要 写 太 多 代码 ，5| 入 错误 的 可 能 性 也 就 随 
之 减少 了 。 
这 同样 有 益 于 设计 的 灵活 性 。 如 果 我 们 需要 更 改 程序 的 运行 结构 ， 就 
可 以 修改 某 一 部 分 ， 而 不 是 查 志 整修 程序 ， 在 各 处 修 修补 补 。 
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REM RABE FAMA. BAER, PRE ALE ARES E 
用 其 他 代码 ， 把 信息 集中 运用 好 ， 那 么 设计 也 就 更 好 。 在 这 一 领 
域 ， 你 同样 可 以 真正 用 目 己 的 聪明 才智 为 编程 创造 价值。 
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现在 我 们 知道 了 ， 如 采 软 件 一 直 不 变化 ， 就 可 以 彻底 避免 出 现 新 错 
误 。 然 而 ， 软 件 必 定 古 要 变化 的 ， 尤 其 是 需要 增加 新 功能 上 时， 变化 是 
不 能 避免 的 ， 所 以 “一 直 不 变化 ”并 不 是 彻底 杜绝 错误 的 办 法 。 


第 6 章 讲 过 ， 如 果 希 望 避 免 代码 出 现 新 错误 ， 可 行 的 办 法 之 一 就 是 把 
变化 的 规模 限定 在 小 光 围 内 。 不 过 ， 如 果 既 要 做 很 多 修改 ， 又 希望 这 
些 变 化 不 要 引入 错误 ， 还 可 以 用 上 另 一 条 法 则 。 它 不 仅仅 用 来 消除 错 
误 ， 还 可 以 保持 程序 的 可 维护 性 ， 降 低 新 增 功 能 的 难度 ， 让 代码 更 容 
易 理 解 。 这 就 是 简洁 定律 (Law of Simplicity): 


软件 任何 一 部 分 的 维护 难度 ， 反 比 于 该 部 分 的 简洁 程度 。 
换 句 话说 ， 杀 一 部 分 的 代码 越 便 滞 ， 未 来 进行 变化 的 难度 就 越 低 。 完 
全 宵 除 维护 的 难度 是 不 可 能 的 ， 但 这 正 是 我 们 要 和 争取 实现 的 目标 : 如 
朱 要 进行 彻 展 的 变化 ， 或 者 新 增 大 量 代 码 ， 不 应 该 遇 到 多 少 困 难 。 


你 可 能 注意 到 了 ， 这 条 法 则 并 不 关心 整个 系统 的 简洁 性 ， 只 是 谈 到 了 
各 个 部 分 的 向 车 性 。 这 和 是 为 什么 呢 ? 


原因 在 于 ， 一 般 的 计算 机 程序 已 经 足够 复杂 了 ， 没 有 人 可 以 一 次 性 全 
面 理解 它 ， 大 家 都 只 能 分 部 份 逐 步 了 解 。 虽 然 程序 里 大 都 有 些 庞大 蒜 
AREA, MULA RAS 要 紧 的 和 证， 在 我 们 疯 读 代码 时 ， 应 该 可 以 理 
解 这 些 庞大 埃 杂 的 结构 。 这 些 部 分 越 向 洁 ， 就 越 容易 被 普通 人 理解 。 
这 一 点 非 党 重要， 尤其 是 你 把 目 己 的 代码 转交 给 其 他 人 ， 或 者 脱离 上 自 
己 的 代码 几 个 月 再 重新 上 手 时 ， 更 是 如 此 。 


用 建筑 结构 来 类 比 
假设 要 搭建 30 英 尺 高 的 钢铁 建筑 。 你 可 以 用 很 多 短 撑 杆 ， 也 就 是 小 
零件 来 搭建 ; 也 可 以 先生 产 3 个 巨大 而 复杂 的 构件 ， 再 把 它们 组 合 
ER, 
如 果 用 短 撑 杆 来 搭建 ， 零 件 是 很 容易 生产 也 容易 买 到 的 。 假 如 某 个 
零件 坏 抒 了 ， 也 很 容易 找到 备件 来 替换 。 整 个 建筑 过 程 是 简单 的 ， 
维护 也 简单 。 
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如 果 采 用 3 个 大 构件 ， 就 需要 花费 大 量 精 力 来 细心 定制 。 因 为 每 个 
构件 都 很 大 ， 因 此 很 难 找 到 并 修正 它 的 缺 隐 。 而 且 ， 如 果 建 筑 完工 
之 后 发 现 构 件 上 有 若干 问题 ， 你 也 不 能 换 掉 它 一 一 抽 挥 任何 一 个 枸 
件 ， 整 个 结构 就 会 倒塌。 所 以 你 只 能 依赖 丑陋 的 补丁 ， 并 且 祈 襟 整 
个 结构 不 会 出 问题 。 


软件 的 情况 与 之 类 似 。 如 果 你 的 代码 写 得 简洁 、 和 上 自 洽 ， 那 么 修正 问 
题 、 维 护 系统 就 很 容易 。 如 果 你 设计 的 是 处 大 党 杂 的 模块 ， 那 么 每 
一 部 分 都 需要 花费 大 量 精 力 ， 也 很 难 做 到 足够 精致 。 这 样 的 系统 就 
很 难 维 护 ， 而 且 必 须 经 常 打 补丁 才 可 以 正常 运行 。 


那么 ， 为 什么 会 有 人 编程 时 要 采用 让 大 繁杂 的 模块 ， 而 不 采用 小 巧 
简洁 的 模块 呢 ? 在 软件 第 一 次 编写 时 ， 采 用 大 模块 表面 上 可 以 闻 
省 下 相当 多 的 时 间 。 如 果 使 用 众多 小 模块 ， 就 必须 花 很 多 时 间 米 组 
合 。 如 果 采 用 大 模块 则 不 会 这 样 ， 零 件 很 少 ， 而 且 很 容易 对 楼 。 


不 过 ， 由 大 模块 构成 的 系统 ， 质 量 要 差 得 多 ， 而 有 全， 将 来 你 得 花 很 
多 时 间 去 修正 错误 ， 结 果 就 是 维护 的 难度 越 来 越 亢 。 相 有 反 ， 简 单 系 
统 的 维护 难度 会 越 来 越 低 。 长 期 来 看 ， 能 保证 效率 的 恰恰 是 简单 的 
系统 ， 而 不 是 复杂 的 系统 。 





那么 ， 在 真正 编程 的 时 候 ， 要 如 何 落实 这 条 法 则 呢 ?” 本 书 的 其 余 章 市 
很 大 篇 幅 都 在 讨论 这 个 问题 。 不 过 总 的 来 说 ， 核 心思 想 就 古 要 让 代码 
中 各 个 部 分 都 尽 可 能 简洁 ， 并 且 尽 力 一 直 保 持 这 种 简洁 性 。 


落实 这 条 法 则 的 一 个 好 办 法 ， 就 是 第 5 章 讲 解 的 潮 进 式 开 发 和 设计 方 
法 。 因 为 每 次 添加 功能 之 前 都 有 个 “重新 设计 ”的 过 程 ， 所 以 系统 能 
持续 简化 。 即 便 不 用 这 种 方法 ， 你 也 可 以 在 增添 新 功能 之 前 ， 伦 点 时 
间 去 化 向 任何 让 你 或 你 的 同事 觉得 不 够 向 蔽 的 代码 。 

无 论 如 何 ， 你 的 代码 一 般 总 是 归 你 负责 ， 化 简 也 归 你 负责 一 一 你 不 能 
奢望 最 初 的 设计 永远 都 是 最 恰当 的 。 在 新 的 形势 和 需求 面前 ， 你 必须 
不 断 重新 设计 系统 的 各 个 部 分 。 
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没 错 ， 这 个 任务 相当 困难 。 不 可 能 任何 时 候 都 有 简单 的 工具 来 写 程 
序 一 一 编程 语言 是 复杂 的 ， 计 算 机 本 身 也 征 复 杂 的 。 不 过 ， 我 们 还 征 
hy 4S DB ATE 


7.1 简洁 与 软件 设计 方程 式 

虽然 你 可 能 已 经 意识 到 了 ， 这 里 还 是 要 说 ， 人 向 竺 定律 告诉 我 们 : 目前 
可 行 的 、 能 够 降低 软件 设计 方程 式 中 维护 成 本 的 最 重要 的 事情 ， 就 是 
把 代码 变 简 洁 。 我 们 不 必 有 预测 未 来 ， 完 全 可 以 只 审视 目 己 的 代码 ， 如 
果 它 足够 复杂 ， 就 立刻 动手 简化 它 。 这 就 是 随时 间 推 移 降低 维护 成 本 
的 办 法 一 一 持续 不 断 地 让 代码 变 得 更 何 笑 。 


进行 这 种 简化 有 一 定 的 工作 量 ， 但 是 总 的 来 看 ， 对 人 向 车 系统 做 修改 总 
是 比 复杂 系统 要 容易 很 多 ， 所 以 ,现在 花 一 点 时 间 去 追求 何洁 ， 将 来 
就 可 以 慷 省 大 量 的 时 间 。 

随 着 系统 维护 成 本 的 降低 ， 各 种 想 做 的 修改 的 可 行 性 也 会 增加 。( 如 
果 你 希望 重 温 细节 ， 请 翻 到 第 4 章 看 看 软件 设计 的 方程 式 。) Lat 
码 可 以 降低 维护 成 本 ， 也 就 增加 了 各 种 可 能 修改 的 可 行 性 。 


7.2 简洁 是 相对 的 

没 错 ， 我 们 希望 把 代码 变 简 洁 。 不 过 如 何 定义 “人 简洁” ， 却 取决 于 你 
的 目标 受众 。 你 觉得 革 个 东西 人 简洁， 你 的 同事 却 可 能 不 这 样 认 为 。 男 
外 ， 你 可 能 觉得 自己 创造 的 某 样 东 西 很 “简洁 ”， 因 为 你 了 解 得 非常 
透彻 。 但 是 在 完全 没 接触 过 的 人 眼 里 ， 它 可 能 无 比 复杂 。 


如 果 你 希望 知道 第 一 次 看 你 代码 的 人 有 什么 看 法 ， 不 妨 找 些 从 没 接触 
过 的 代码 来 看 看 。 你 需要 理解 的 不 只 是 那 一 行 行 的 代码 ， 而 古 整 个 程 
序 在 干什么 ; 而 且 你 还 必须 弄 清 楚 ， 要 做 修改 该 如 何 动 手 。 基 他 人 看 
你 的 代码 的 时 候 ， 就 是 这 种 感觉 。 你 可 能 发 现 了 ， 如 采 阅 读 别 人 写 的 
程序 ， 就 算 不 很 复杂 的 代码 也 会 让 人 相当 郁 况 。 


所 以 ， 一 个 不 错 的 做 法 就 古 在 目 己 的 代码 注释 中 加 上 类 似 “ 头 一 次 看 
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这 段 代码 ? 那么 ……” 这 样 的 注释 ， 在 其 中 给 出 一 些 概略 的 解释 ， 帮 
助 其 他 人 理解 。 写 注释 时 应 当 考 虚 到 ， 读 者 完全 不 了 解 这 段 程序 ， 因 
为 如 果 是 你 初次 接触 某 些 东 西 ， 你 全 怕 也 对 它 一 无 所 知 。 


无 数 的 软件 项 目 在 这 方面 表现 得 一 团 精 。 翻 开 写 给 开发 者 的 文档 ， 你 
Hea PLACE ER, TRAE AS. MAH RARER MR 
目 上 工作 了 很 长 时 间 ， 事 情 就 非常 简单 ， 因 为 这 张 有 大 其 链接 的 索引 
表 ， 可 以 帮 他 们 迅速 找到 要 看 的 内 容 。 但 是 ， 这 种 文档 对 新 手 来 说 非 
常 复杂。 而 对 工作 了 很 长 时 间 的 开发 人 员 来 说 ， 如 果 在 文档 中 新 增 一 
页 ， 在 上 面 用 简单 醒目 的 按钮 取代 原本 熟悉 的 链接 ， 只 会 把 事情 搞 复 
江 ， 因 为 他 要 做 的 只 是 在 文档 中 迅速 找到 相应 的 资料 。 


比 起 营 杂 的 文档 ， 更 糟糕 的 情况 就 是 没有 文档 ， 开 发 软件 的 人 认为 其 
他 人 可 以 日 己 找 出 问题 ,或 者 “ 早 就 知道 ”程序 是 怎样 运行 的 。 对 开 
发 人 员 来 说 ， 目 己 写 的 程序 的 原理 是 显而易见 的 ， 但 对 其 他 人 来 说 ， 
这 和 十 一 个 完全 阳 生 的 世界 。 


应 用 场合 (上下文) 也 十 非常 重要 的 。 在 程序 代码 中 ， 人 先进 技术 如 果 
使 用 得当 ， 通 第 会 让 代码 出 话 。 但 是 ， 如 朱 程 序 的 复杂 内 部 结构 只 有 
BIT Web 页 才能 看 到 ， 其 他 方式 部 不 可 行 ， 这 束 算 不 上 向 活 了 ， 
AMENA A A AC, MARA. 


AR, SET A SP BK SE RAY ARP, BRT oe Fy Se 
A, FERPA rhe bth A Fe dk i RAY LEA AY 
司机 根本 来 不 及 去 看 那么 多 文字 ， 所 以 这 么 做 很 思春 。 但 是 在 程序 使 
用 手册 里 ， 给 出 大 量 的 解释 性 文字 就 比 一 句 话 的 概要 挡 述 向 菩 得 多 。 
所 以 ， 本 书 的 每 章 并 不 会 一 句 话 就 完事 。 FRESIA 
FE, IPA ET 


ESA fel A A ee FS, IB Ter de RA MEME? 不 ， 绝 对 
不 是 。 我 们 做 的 每 一 件 事 都 会 有 目标 受众 ， 每 个 应 用 场合 通常 都 存在 
诸多 限制 。 所 以 ， 追 求 而 洁 古 绝对 可 行 的 。 重 要 的 十 要 在 设计 软件 时 
考虑 到 上 述 这 些 因 系 ,这样 其 他 人 在 真正 使 用 时 就 会 感到 软件 是 非常 
出 单 的 。 
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完成 某 项 任务 的 最 好 用 的 工具 是 什么 ， 在 软件 开发 的 世界 里 ， 这 是 
个 争论 不 休 的 问题 。 各 人 偏爱 的 文本 编辑 器 不 同 ， 编 程 语言 不 同 ， 
操作 系统 也 不 同 。 在 软件 开发 中 最 著名 的 “战争 ”， 大 概 就 是 两 款 
文本 编辑 器 之 间 的 战争 ， 这 两 款 编辑 器 就 是 Vi 和 Emacs。 每 一 派 的 用 
户 都 宣称 ， 自 己 喜欢 的 编辑 器 从 根本 上 讲 要 优 于 另 一 款 编 辑 器 ， 


其 实在 编写 代码 这 件 事 上 ， 基 本 没什么 工具 能 从 根本 上 强 过 另 一 
款 。 实 际 的 情况 是 ， 特 定 的 用 户 觉得 用 某 些 工具 更 容易 解决 手头 的 
具体 问题 ，Emacs 用 户 认 为 Emacs 是 最 简洁 的 编程 工具 ， 而 Vi 的 用 户 
认为 vi 是 最 简洁 的 。 从 某 种 程度 上 讲 ， 之 所 以 会 有 这 样 的 局 面 ， 是 
因为 人 的 喜好 是 有 巨大 差别 的 ， 大 家 喜欢 的 工作 方式 不 同 ， 思 维 方 | 
ALA, AROMA DARE, AMARA TE. PARE 
| 入 一 点 来 看 ， 工 具 的 简洁 程度 是 与 每 个 人 的 使 用 熟练 程度 相关 的 。 

| 任何 人 ， 只 要 使 用 某 款 工具 足够 长 的 时 间 ， 都 会 更 热 三 它 ， 从 个 人 | 
| 角度 出 发 ， 会 觉得 它 比 其 他 工具 更 简洁 。 如 果 要 求 一 款 新 工具 表现 
得 同样 阐 单 ， 那 么 这 款 工 具 必须 简洁 到 极致 。 显 然 程序 页 用 的 文本 
编辑 器 做 不 到 这 一 点 。 


不 写 程序 的 人 可 能 觉得 两 款 编辑 器 都 太 复杂 了 ， 已 经 无 法 理解 ， 这 
个 例子 再 次 说 明 ， 简 洁 是 相对 的 。 








fa 


如 果 工 具 不 适用 于 手头 的 某 个 任务 ， 或 者 在 设计 中 做 了 错 
误 的 选择 (参见 8.2 节 )， 此 工具 就 可 能 带 来 问题 。 如 果 忽 
略 这 些 问题 ， 工 具 的 相对 简洁 性 就 在 于 ， 在 某 个 具体 的 情 
兄 下 ， 程 序 员 可 以 自行 选择 用 起 来 最 舒服 的 工具 。 








+ 





73 简洁 到 什么 程度 ? 


一 旦 开始 做 某 个 项 目 ， 就 会 遇 到 关于 简洁 的 问题 。 我 们 要 简洁 到 什么 
需要 把 事情 简化 到 什么 程度 ? 它 已 经 足够 简洁 了 吗 ? 


+ 的 。 不 过 即便 如 此 ， 还 是 存在 更 简洁 和 更 复杂 的 差 






别 。 从 用 户 的 角度 出 发 ， 你 的 产品 可 能 很 难 用 ， 也 可 能 很 容 多 使 用 ， 
还 可 能 介 于 两 者 之 间 。 同 样 道理 ， 在 其 他 程序 员 看 来 ， 你 的 程序 阅读 
起 来 也 可 能 比较 困难 ， 或 者 比较 简单。 


那么 ， 要 简洁 到 什么 程度 呢 ? 
要 听 真 话 吗 ? 

你 真 的 想 要 成 功 ? 
简单 到 傻子 也 能 懂 | 


关于 简洁 程度 ， 有 一 点 相当 有 意思 : 在 大 多 数 场合 ， 任 何 普通 人 可 以 
用 到 的 ， 天 才 也 可 以 用 。 所 以 ， 软 件 的 可 能 用 户 的 类 型 ， 或 许 比 设想 
的 要 多 得 多 。 

但 是 大 家 通 贡 不 理解 ， 怎 么 做 才 算 真正 简单 到 傻子 也 能 伸 。 全 个 例 
子 : 大 商场 里 都 有 一 张 示 意图 标识 方位 。 在 最 清楚 的 示意 图 上 会 有 一 
个 大 红 点 ， 并 在 你 面前 用 醒目 字体 标识 “你 在 这 里 。 精 糙 一 点 的 示 
意图 中 间 会 有 一 个 不 易 发 现 的 小 黄 三 角形 ， 在 示意 图 侧 边 用 文字 说 明 
“小 黄 三 角形 表示 “你 在 这 里 “。 这 只 会 让 看 示意 图 的 人 更 困 误 ， 你 
可 能 需要 化 五 六 分 钟 才 能 找到 想 去 的 地 方 。 


对 设计 示意 图 的 人 来 说 ， 出 现 这 种 情况 并 不 难 理解 。 既 然 花 了 大 量 的 
时 间 来 设计 ， 那 么 对 他 来 说 ， 这 张 示意 图 显然 足够 重要 ， 他 乐意 花 几 
分 钟 来 查看 和 学 习 ， 找 到 自己 想 要 的 信息 。 但 是 对 其 他 人 想 的 只 是 看 
看 这 张 图 ， 所 以 根本 不 关心 花 了 多 少时 间 来 设计 。 用 户 只 希望 尽 可 能 
简单 ， 这 样 才 能 迅速 上 手 ， 才 能 真正 用 起 来 。 

许多 程序 员 在 这 方面 做 得 尤其 差劲 。 他 们 以 为 别人 都 愿意 花 很 多 时 间 
来 学 习 自己 写 的 代码 ， 毕 竞 这 是 自己 花 很 多 时 间 写 出 来 的 。 这 些 程序 
员 很 重视 自己 的 代码 ， 所 以 对 其 他 人 也 应 当 同 样 重视 。 

没 错 ， 程 序 员 通常 都 是 非常 聪明 的 人 。 但 是 “ 噢 ， 其 他 程序 员 会 理解 
我 写 的 所 有 代码 ， 没 必要 简化 或 者 注释 ”的 看 法 仍然 是 不 对 的 。 这 个 
问题 与 智商 无 关 ， 而 与 背景 知识 有 关 。 第 一 次 接触 你 代码 的 程序 员 完 
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全 没有 任何 背景 知识 ， 他 必须 学 习 。 学 习 的 难度 越 低 ， 找 出 问题 的 束 
度 也 就 越 快 ， 使 用 起 来 也 越 容易 。 


降低 代码 学 习 难度 的 方法 有 很 多 : 简洁 的 注释 ， 简 单 的 设计 ， 循 序 源 
进 的 引导 ， 等 等 。 


不 过 ， 如 果 你 的 代码 没有 做 到 傻子 都 能 看 懂 ， 其 他 人 学 起 来 就 会 遇 到 
困难 。 他 们 会 误解 ， 会 制 霹 bug, SSM AM. HA DA 
生 的 时 候 ， 他 们 会 找 谁 ? 对 ， 就 是 你 。 这 时 候 你 就 得 花 时 间 回 和 葵 他 们 
的 各 种 问题 。( 嗯 ， 了 听 起 来 挺 讽刺 的 ， 对 吧 ? ) 


我 们 谁 也 不 希望 被 当成 僚 子 来 训导 或 对 待 。 所 以 有 时 候 我 们 要 摘出 些 
稍微 复杂 点 的 东西 ， 这 样 才 显得 比 用 户 或 其 他 程序 员 更 聪明 。 我 们 压 
夸 其 谈 ， 故 意 把 软件 做 复杂 点 ， 让 别人 膜拜 我 们 的 智商 ， 让 他 们 因为 
自己 搞 不 懂 而 感到 思春 。 他 们 可 能 觉得 自己 永远 也 不 会 有 我 们 那么 聪 
明 ， 这 确实 让 我 们 有 种 成 就 感 。 可 是 说 真 的 ， 这 样 做 能 帮助 他 们 吗 ? 


相反 ， 如 果 你 的 产品 或 代码 简单 到 傻子 都 懂 ， 大 家 就 都 可 以 理解 它 。 
这 样 ， 其 他 人 会 感觉 自己 是 聪明 的 ， 也 可 以 完成 自己 希望 完成 的 任 
务 ， 同 时 你 并 没有 增加 多 少 负担 。 其 实 ， 如 果 你 把 事情 做 得 简单 而 不 


ER, KRKBREREM. 


当然 ， 你 的 家 人 没 必要 能 读 懂 你 的 代码 。 所 以 ， 简 单 仍然 是 相对 的 ， 
你 的 代码 的 目标 受众 不 是 家 人 ， 而 是 其 他 程序 员 。 但 对 这 些 程序 员 来 
说 ， 你 的 代码 应 该 尽 可 能 简洁 ， 容 易 理解 。 编 程 过 程 中 完全 可 以 使 用 


所 需 的 各 种 先进 技术 来 达到 这 种 简洁 ， 但 是 代码 本 身 必 须 是 简洁 的 。 


面 对 “ 要 做 到 多 简洁 ”这 种 问题 时 ， 你 可 能 需要 同时 间 目 己 : RR 
竟 是 要 让 用 户 理解 ， 感 到 快乐 ， 还 是 让 他 们 困 取 ， 感 到 诅 形 ? ”如 果 
CPR A, WARM RRR AEE: 简单 到 余子 部 能 履 ， 


74 ”保持 一 到 


要 做 到 简单 ， 保 持 一 致 是 很 重要 的 工作 。 如 果 你 在 一 个 地 方 及 用 了 荣 
种 规则 ， 就 应 当 在 其 他 每 个 地 方 都 遭 守 这 种 规则 。 
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如 果菜 个 变量 命名 为 somethingLikeThis， 那 么 所 有 的 变量 
都 应 该 遵守 这 种 规则 (其 他 变量 名 应 该 是 otherVariable, 
anhtoerNameLikeThat 等 )。 如 果 变 量 命 名 为 named like this, 
那么 所 有 的 变量 都 应 该 用 小 写 ， 并 且 用 下 划 线 连接 这 些 单 词 。 


如 来 代码 不 能 保持 一 致 ， 程 序 员 理解 和 阅读 起 来 都 要 更 加 困难 。 
还 可 以 华 目 然 语 言 的 例子 说 明 这 个 进 理 ， 看 下 面 两 句 话 : 


¢ This is a normal sentence with normal words that everybody can understand. 
¢ tHisisanOrmalseNtencewitHnorMalwordsthAtevErybOdycAnunderStaNd. 


两 句 话 的 意思 相同 ,但 是 常 一句 更 容 多 阅读 ， 因 为 它 件 合 大 家 的 书写 习 
i. SK, 第 二 句 也 不 古 看 不 懂 ， 但 十 你 真有 的 希望 整 本 书 虱 写成 这 个 样 
子 吗 ? 同样 的 道理 ， 如 采 茶 个 程序 毫 无 一 致 性 可 言 ， 你 愿意 去 看 吗 ? 


在 编程 时 ， 有 时 候 你 做 什么 并 不 重要 ， 只 要 一 直 保 持 相同 的 方式 去 做 
即 可 。 理 论 上 说 ， 你 当然 可 以 把 代码 写 得 无 比 复杂 ， 但 十 也 要 保持 复 
杂 的 一 致 性 ， 这 样 其 他 人 才 可 以 阅读 。( 当 然 ， 更 好 的 办 法 是 保持 一 
致 并 且 做 到 人 简单， 但 是 如 采 达 不 到 极其 向 单 ， 至 少 要 保持 一 致 。) 


在 许多 情况 下 ， 完 全 保持 一 致 可 以 让 编程 更 简单 。 如 果 程 序 中 每 个 对 
ZAAPA — name 字段 ， 你 就 可 以 写 一 小 段 向 单 代 码 来 处 理 整 个 程序 
中 所 有 对 象 的 name 字段 。 但 是 如 果 在 对 象 A 中 它 叫 做 a_name， 在 
AR BH EM name of mine， 你 就 得 为 对 象 A 和 对 象 B 分 别 做 
特殊 处 理 。 


同样 ， 程 序 的 内 部 行为 应 当 保 持 一 致 。 如 果 你 已 经 熟悉 了 系统 的 某 个 
apa, BL A] DAR AR Bota, AA eh AA ae 
类 似 的 。 如 来 要 使 用 A 部 分 ， 程 厅 员 需要 调用 三 个 图 数 ， 骨 写 一 些 代 
fh; 那么 为 了 使 用 B 部 分 ， 应 当 调 用 和 那 三 个 国 数 类 似 的 几 个 国 数 ， 
再 写 一 些 代 码 。 如 朱 在 A 部 分 中 ， 有 个 叫 dump 的 程序 可 以 打印 出 所 
有 内 部 变量 ， 那 么 B 部 分 中 的 dump 函数 应 当 完 成 同样 的 功能 。 千 万 
别 让 程序 员 每 次 遇 到 新 的 部 分 都 重新 学 习 。 
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真实 世界 里 或 许 不 存在 这 样 的 一 致 性 ， 但 是 程序 的 世界 由 你 负责 ， 所 
以 必须 保持 程序 的 简单 和 一 致 。 


在 真实 世界 中 也 有 些 一 致 性 的 例子 。 在 亚洲 大 部 分 国家 ， 人 们 用 筷子 
吃饭 。 在 欧美 国家 则 使 用 刀 又 。 昌 然 有 两 类 餐具 ， 但 是 总 的 来 看 ， 在 
某 个 地 区 还 是 相当 一 致 的 。 假 设 每 次 你 去 别人 家 ， 都 必须 学 会 使 用 新 
的 餐具 ，Bob 家 用 的 是 剪刀 ，Mary 家 用 的 是 硬 纸板 ， 这 样 吃 羽 就 成 了 
Aal, 1128? 


AHEbLaH— AZ — At, RAZA. ASR, MAIER 
A, BUESA AAA, FR le te Se BY 理 
杂 性 ， 就 不 必 再 进行 重复 劳动 。 


75 可 读 性 
软件 开发 领域 反复 强调 一 点 : 代码 被 阅读 的 次 数 远 多 于 编写 和 修改 的 
次 数 。 所 以 ， 保 证 代码 容易 阅读 很 重要 。 


代码 可 读 性 主要 取决 于 字母 和 符号 之 间 的 空白 排 布 。 


如 果 世 界 是 一 团 漆黑 ， 就 分 不 出 任何 物体 ， 因 为 万 物 都 是 黑色 的 一 
团 。 同 样 道理 ， 如 果 整 个 文件 的 代码 乱 成 一 团 ， 没 有 格式 稳定 、 符 合 
淄 辑 的 空白 ， 就 很 难 拆 分 各 个 部 分 。 要 把 各 部 分 拆 分 开 来 ， 就 必须 留 


出 空 日 。 


空白 太 多 是 不 必要 的 ， 因 为 这 样 很 难 发 现 事物 之 间 的 联系 。 空 白 太 少 
也 不 必要 ， 因 为 这 样 很 难 分 解 。 代 码 到 底 应 该 留 出 怎样 的 空白 ,没有 
什么 硬性 的 、 直 接 的 规则 ， 唯 一 的 规则 是 ， 代 码 之 间 留 出 的 空白 应 当 
保持 一 致 规范 ， 空 白 应 当 能 有 助 于 读者 理解 代码 的 结构 。 











pl 








FA: SAHA 


下 面 的 代码 很 难看 懂 ， 因 为 空白 太 少 了 ， 几 平 没 有 提供 关于 代码 
结构 的 信息 。 
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X=14+2 ; y=34+4; z=x+y; if (z>y+x) {print"error"; } 


同样 的 代码 ， 这 里 的 空白 又 太 多 了 ， 读 者 很 难 理解 代码 的 结构 。 


x = 1+ dl 
Y = 3 +4 
Z = X 4 y 
ıf (z > Y+xX) 
{ print "error" ; 


写成 上 面 这 样 ， 黄 至 比 没有 留 出 空白 的 代码 还 要 难看 懂 。 
下 面 这 段 代码 的 空白 设置 得 比较 合适 : 


“= 1 + 2; 
y=3 + 4; 
za x 4 


Yi 
if (z > y +x) 1 
print "error"; 
| 


这 样 读 起 来 就 容易 多 了 ， 也 有 助 于 大 家 理解 程序 员 写 这 段 代码 的 目 
的 。 首 先 给 3 个 变量 赋值 ， 然 后 根据 某 个 条 件 ， 显 示 某 个 错误 。 程 序 
员 通 过 安排 合适 的 空白 ， 把 代码 的 远 辑 结构 清楚 地 向 读者 表达 出 来 。 
如 果 代 而 很 容易 阅读 ， 也 就 容易 修正 。 在 前 面 的 例子 中 ， 如 果 空 白 
留 得 合适 ， 我 们 可 以 很 容易 地 发 现 ，z 了 永远 不 可 能 比 v + x 大 ， 因 为 
z 了 永远 等 于 yY + x. AYU, if (z > y + x) 开头 的 这 段 代 码 其 实 
是 没有 用 的 ， 应 当 删 掉 。 

一 般 来 说 ， 如 果 某 段 代 码 有 很 多 bug， 又 难以 阅读 ， 那 么 首先 要 做 
的 是 让 它 更 容易 阅读 。 然 后 ，bug 在 哪里 才能 看 得 更 清楚 


7.5.1 命名 
可 读 性 的 另 一 部 分 重要 内 容 是 为 变量 、 函 数 、 类 等 选择 合适 的 名 字 ， 
理想 的 命名 应 该 这 样 ， 


名 字 应 当 足 够 长 ， 能 够 完整 表达 其 意义 或 描述 其 功能 ,但 
不 能 太 长 ， 以 免 影响 阅读 。 


Alt, QM 4A ae, TRENNEN, FERS AA 
Ff, ERASRAG m1? 比如 ， 如 采 有 个 国 数 只 会 在 某 
一 行 由 你 目 己 调用 一 次 (而 且 这 一 行 没 有 其 他 代码 )， 它 的 名 字 可 以 
很 长 。 不 过 ， 如 果菜 个 函数 会 经 第 在 复杂 的 表达 式 中 用 到 ， 大 概 就 该 
取 个 短 一 些 的 名 字 AERC, RENA END GE) 


示例 : 命名 
这 段 代码 的 命名 就 很 糟糕 : 
Y = 5143, E, ti; 
pla); 
这 些 名 字 没 有 说 明 变 量 的 用 途 和 函数 的 功能 。 
这 段 代 码 与 之 相同 ， 但 命名 很 不 错 : 
quarterly total = sum(january, february, march); 
print (quarterly total); 
还 是 同样 的 代码 ， 但 名 字 太 长 ， 难 以 阅读 : 


quarterly total for company in 2011 as of today = 

add all of these together and return the result (january_ 

total amount, 

february total amount, march total amount); 

send to screen and dont wait for user to 

respond (quarterly total for company in 2011 as of today); 
这 些 名 字 占 据 了 太 多 空间 ， 很 难看 明白 。 所 以 从 某 个 角度 来 看 ， 问 
题 又 回 到 了 字母 和 符号 之 间 应 该 怎样 留 出 空白 。 


7.5.2 ¡EE 


为 体 证 代码 的 可 谈 性 ， 好 的 的 往 释 也 很 重要 。 但 是 ， 代 码 的 意图 通 前 
不 应 该 用 注释 来 说 明 ， 直 接 阅 读 代码 就 应 当 能 够 理解 。 如 果 发 现 意图 
不 够 明显 ， 那 么 就 说 明 这 段 代 码 还 可 以 变 得 更 侧 单 。 如 末 你 的 代码 实 
在 不 能 更 向 单 ， 才 应 该 写 注释 来 说 明 。 


注释 的 真实 目的 ， 是 在 理由 不 够 清晰 明显 时 加 以 解释 。 如 琳 不 解释 ， 
其 他 程序 员 在 修改 这 段 代码 时 可 能 会 很 困惑 ， 如 未 不 明日 这 些 理由 ， 
他 们 可 能 会 删改 其 中 重要 的 部 分 。 

有 些 程 序 员 相信 ， 可 读 性 是 追求 代码 简洁 性 的 全 部 和 终极 目标 ， 如 果 
写 的 代码 很 容易 阅读 ， 你 就 已 经 做 完了 设计 师 要 做 的 一 切 。 但 事实 并 
非 如 此 ， 你 的 代码 可 能 很 容易 阅读 ， 但 系统 仍然 非常 复杂 。 不 过 ， 保 
证 代码 的 可 读 性 是 非常 重要 的 ， 而 且 它 往往 是 通 往 优 秀 议 计 之 路 上 应 
当 走出 的 第 一 步 。 


7.6 简洁 离 不 开设 计 

不 幸 ， 没 有 人 天 生 就 会 构建 简洁 的 系统 。 如 果 设计 师 不 倾注 精力 ， 系 
统 就 会 逐渐 变 成 杂乱 庞大 的 怪物 。 

如 果 项 目 没 有 好 的 设计 ， 而 且 持 续 膨 胀 ， 最 终 的 局 面 会 复杂 得 让 你 头 
疼 。 有 些 人 可 能 想象 不 出 来 这 种 场景 ， 就 像 有 些 人 预计 不 到 午饭 后 会 
发 生 什 么 ， 或 者 有 些 人 缺乏 足够 的 经 验 来 理解 最 终 的 复杂 局 面 。 还 有 
些 企业 的 文化 是 “ 噢 ， 我 们 只 不 过 缺 几 项 新 功能 ;做事 征 应 当 规 范 
As 然而 …… 但 是 …… 可 惜 …… ”不 过 ， 总 有 一 天 你 的 项 目 会 失败 。 
无 论 你 可 以 找到 多 少 理由 ， 失 败 是 避免 不 了 的 。 

反 过 来 ， 如 果 你 的 设计 非常 好 ， 一 般 却 难得 听 到 多 少 称赞 。 设 计 中 的 
缺陷 是 大 家 都 看 得 到 的 ， 但 逐步 演变 为 良好 设计 的 改进 过 程 ， 却 是 不 
就 悉 代 码 的 人 看 不 到 的 。 于 是 ， 设 计 师 就 成 了 一 个 费力 不 讨好 的 工 
(E, FERRARA i meio AS, (Bae RPA AA NE, ¿ 
有 人 会 注意 到 。 

所 以 ， 计 这 本 书 来 表扬 你 吧 。 关 于 设计 ， 你 有 设 有 多 加 思考 ? A? 于 
的 不 错 ! 你 的 用 户 和 同事 会 感觉 到 好 处 一 王 运 行 琉 畅 的 软件 ， 按 时 的 
发 布 ， 条 理 分 明 、 容 易 理解 的 代码 。 你 对 目 己 的 工作 有 是 人 够 的 信心 ， 
完工 之 后 充满 了 成 就 感 。 其 他 程序 员 知 道 要 花 多 少 精力 才 让 一 切 有 条 
理 吗 ? 可 能 不 知道 。 但 是 ， 这 不 要 紧 。 除 去 你 周围 人 的 称赞 ， 还 有 其 
他 的 奖励 。 
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只 有 在 个 别 情况 下 ， 你 的 工作 才 会 赢得 其 他 人 的 称赞 。 但 古 别 绝望 ， 


最 后 总 会 有 人 注意 到 的 。 在 这 之 前 ， 你 应 当 享受 有 将、 正确 的 设计 所 
带 来 的 各 种 积极 结 米 。 


a . 
An 
tar mn 
+ 


如 果 把 本 书 中 提 到 的 设计 原则 应 用 到 手头 的 项 目 中 ， 经 验 
不 多 的 程序 员 或 同事 可 能 要 花 很 长 的 时 间 才 能 理解 ， 他 们 
为 什么 同样 需要 追求 优秀 的 设计 。 请 他 们 来 读 这 本 书 可 能 
会 帮 上 点 忙 ， 如 果 他 们 不 能 或 不 愿意 ， 应 当 持 续 引 守 他 们 
(如 果实 在 不 得 已 ， 就 强迫 他 们 /追求 优秀 的 设计 ， 他 们 会 
在 (最多) 几 年 后 认识 和 到， 优秀 的 设计 能 带 来 怎样 的 好 处 。 
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75 











身 为 职业 程序 员 ， 你 很 可 能 听 说 过 (或 者 经 历 过 ) FANERFRE 
2: 五 年 前 项 目 局 动 时 ， 这 项 技术 还 是 很 先进 的 ， 可 和 古 现 在 过 时 了 。 
因为 技术 已 经 被 淘汰 ， 事 情 越 变 越 复杂 ， 所 以 项 目 完 成 的 锅 望 越 来 越 
Wi. (eM A BK, ARDS LAA.” 


还 有 一 个 情况 也 很 常见 :“ 我 们 的 开发 速度 不 够 快 ， 跟 不 上 现在 用 户 
MER. “我 们 正在 开发 ,，X 公司 就 以 更 快 的 速度 完成 了 比 我 们 更 好 
的 产品 。” 


我 们 现在 知道 ， 这 些 问 题 的 根源 都 在 复杂 性 。 开 始 的 时 候 项 目 古 便 单 
的 ， 只 要 一 个 月 就 能 完成 ;然后 复杂 性 增加 了 ， 于 征 需 要 三 个 月 时 
间 ， 再 然后 ， 每 个 部 分 都 更 加 复杂 ， 所 以 项 目 需要 九 个 月 才能 完成 。 


复杂 性 是 会 又 加 的 ， 而 且 不 是 简单 的 线性 全 加 。 也 就 是 说 ， 下 述 假设 
ENT: 之 前 有 10 项 功能 要 开发 ， 因 此 再 加 1 项 只 会 增添 10% 
的 工作 量 。 因 为 新 增 的 功能 需要 与 已 有 的 10 项 功能 相 协 调 ， 所 以 如 
果 开 发 新 功能 需要 10 小 时 ， 可 能 还 要 花 10 小 时 才能 保证 已 有 的 10 
项 功能 与 新 功能 正常 交互 。 原 有 功能 越 多 ， 新 增 功能 的 成 本 就 越 高 。 
优秀 的 设计 可 以 尽量 避免 此 类 问题 ， 但 是 每 项 新 功能 仍然 会 有 单独 的 
成 本 。 


有 些 项 目 从 一 局 动 吏 设 定 了 称 多 的 需求 ， 所 以 永 还 无 法 发 布 审 一 版 。 
如 妥 遇 到 这 种 情况 ， 就 应 当 删 减 功 能 。 初 次 发 布 不 应 当 设 定 过 高 的 目 
标 ， 而 应 当先 让 程序 跑 起 来 ， 再 持续 改进 。 


除了 新 增 功能 ， 其 他 一 些 做 法 也 会 增加 复杂 性 ， 以 下 列 出 了 最 常见 的 
做 法 。 


1. 扩展 软件 的 用 途 

一 般 情况 下 ， 应 当 绝 对 禁止 这 样 做 。 市 场 部 可 能 巴 望 着 某 款 软件 既 能 
够 计算 个 税 ， 又 可 以 充当 菜谱 ， 这 样 的 需求 ， 你 必须 尽 全 力 抵制 。 软 
件 应 当 坚 守 已 经 确定 的 用 途 ， 只 要 妥善 完成 这 些 目标 ， 你 就 会 获得 成 
功 (前 提 是 该 软件 能 帮 到 用 户 ， 切 实 满 足 其 需求 )。 
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2. 新 增 程序 员 

役 钵 ， 往 图 队 里 增加 新 人 并 不 会 让 事情 变 条 单 ， 相 反 会 更 复杂 。Fred 
Brooks 的 名 作 (A APPT) 说 的 就 是 这 个 道理 。 如 采 已 经 有 了 10 个 
开发 人 员 ， 再 增加 1 个 人 ， 就 意味 着 需要 为 他 设 定 人 台 适 的 六 位 ， 花 时 
回 让 之 前 的 10 个 人 适应 新 人 ， 花 时 间 让 新 人 学 会 与 那 10 个 人 沟通 ， 
如 此 等 等 。 相 比 众多 平庸 的 开发 人 员 ， 少 量 精干 的 开发 人 员 更 容易 获 
得 成 功 。 











3. 做 无 谓 的 改变 

每 做 一 点 改变 ， 都 会 增加 复杂 性 。 无 论 是 需求 变化 、 设 计 变化 ， 或 是 
禾 改 某 段 代码 ， 都 有 可 能 增加 新 的 bug， 另 外 别 忘 了 算 上 决定 如 何 
变化 所 需 的 时 间 ， 实 现 它 所 需 的 时 间 ， 验 证 它 是 否 影 响 到 原 有 系统 所 
需 的 时 间 ， 记 录 它 的 时 间 ， 测 试 它 的 时 间 。 每 做 一 点 新 变化 ， 整 体 复 
杂 性 就 会 增加 一 点 ， 所 以 变化 越 多 ， 每 个 变化 要 花 的 时 间 就 越 长 。 做 
出 某 些 变化 是 重要 的 ， 但 是 应 当 谨 慎 决 策 ， 而 不 古 一 拍 脑 瓜 就 定 了 。 












4. 困 于 糟糕 的 技术 

一 般 来 说 ,“ 困 于 糟糕 的 技术 ” 指 的 是 你 之 前 决定 了 采用 基 种 技术 ， 
因为 极度 依赖 它 ， 长 期 无 法 摆脱 。 这 里 说 的 “ 精 烂 ”"， 意 思 是 你 深 陷 
其 中 (未 来 无 法 简单 地 切换 到 其 他 技术 )， 不 能 灵活 地 适应 未 来 的 需 


求 ， 或 是 达 不 到 设计 向 话 软 件 所 需 的 质量 标 惟 。 








5. 理解 错误 

程序 员 不 理解 自己 的 工作 ， 就 容易 设计 出 复杂 的 系统 。 这 可 能 是 恶性 
循环 ， 理解 错误 导致 复杂 性 ， 复 杂 性 又 进一步 加 剧 理解 错误 ， 如 此 往 
复 。 提 升 设计 水 平 的 最 主要 办 法 是 ， 确 保 自己 完全 理解 所 用 的 系统 和 
工具 。 你 对 它们 的 理解 越 到 位 ， 对 软件 开发 的 一 般 规 律 了 解 越 多 ， 你 
的 设计 就 越 简洁 。 





6. 糟糕 的 设计 或 不 做 设计 
一 般 来 说 ， 它 指 的 是 “没有 为 变化 做 计划 "。 万 物 都 是 会 变化 的 ， 项 目 
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增长 时 ， 设 计 仍 然 要 维持 简单 。 你 必须 一 开始 就 做 好 设计 ， 而 且 在 系 
统 膨胀 时 不 断 进行 优秀 的 设计 ， 否 则 ， 复 杂 性 就 会 迅速 增长 ， 因 为 如 
果 设 计 得 不 好 ， 每 项 功能 都 会 让 代码 加 们 复杂， 而 不 是 只 复杂 一 点点 。 


7. 重新 发 明 轮 子 

如 果 有 相当 不 错 的 现成 协议 ， 还 要 自己 发 明 协 议 ， 那 么 仅仅 为 了 把 软 
件 跑 起 来 ， 这 些 协议 也 会 花 去 你 大 量 的 时 间 。 决 不 要 什么 都 靠 自力 更 
生 ， 去 自己 开发 什么 Web 服务 器 、 协 议 或 者 重要 的 类 库 ， 除 非 它 们 十 
你 的 最 终 产 品 。 只 有 在 满足 以 下 任何 一 个 条 件 的 前 提 下 ， 重 新 发 明 轮 
子 才 有 价值 : 


(2) HAMM “HF” ARE, SERIE; 

(3) 现 有 的 “轮子 ”根本 无 法 满足 你 的 需求 ; 

(4) 现 有 的 “轮子 ”缺乏 良好 的 维护 ， 而 你 也 不 能 接 过 维护 的 任务 (EL 
如 ， 你 没有 产 代码 )。 


这 些 因素 都 会 逐 灶 影 啊 你 的 项 目 ， 但 不 会 在 短 时 间 凸 显 。 它 们 大 多 数 
会 造成 长 期 的 负面 影响 ， 其 至 一 两 年 内 都 觉察 不 出 来 ， 所 以 如 来 有 人 
指出 这 些 问题 ， 通 常 大 家 也 不 觉得 有 什么 坏处 。 即 便 你 走 上 这 条 路 ， 
重新 发 明 的 轮子 可 能 看 来 也 没有 问题 。 但 是 随 着 时 间 的 推移 ， 尤 其 是 
这 些 轮子 堆积 起 来 后 ， 复 杂 性 会 越发 明显 ， 不 断 累 加 ， 节 终 你 就 成 了 
ABS AA RIAA BS, PALA THA So 


8.1 复杂 性 与 软件 的 用 途 

你 正 开 发 的 任何 系统 ， 其 基本 用 途 应 当 相当 简单 。 这 样 开 发 出 来 的 系 
统 ， 既 注 足 实际 需求 ， 整 体 来 说 也 是 简单 的 。 但 是 ， 如 果 你 给 系统 淮 
加 新 功能 去 满足 其 他 目标 ， 事 情 就 立刻 变 复 杂 了 。 举 例 来 说 ， 文 字 处 
理 软 件 的 基本 功能 就 是 帮助 用 户 写 作文 档 。 如 果 突 然 要 求 它 能 够 阅读 
邮件 ， 最 终 就 会 得 到 的 非常 旋 唐 复杂 的 玩意 。 你 能 设想 它 的 用 尸 界 面 
吗 ? 各 个 按钮 都 要 放 在 哪里 ? 我 们 都 知道 ， 这 不 是 文字 处 理 软件 本 来 
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的 用 途 。 这 么 做 ， 其 至 都 不 是 在 扩展 软件 的 用 途 ， 而 是 增添 与 目的 无 
关 的 功能 。 


同样 重要 的 是 要 思考 用 户 的 需求 。 用 户 之 所 以 要 使 用 软件 ， 总 是 有 自 
,的 需求 。 理 想 情 况 下 ， 软 件 的 用 途 应 当 相 当 接近 用 户 的 需求 。 举 个 
例子， 假设 用 户 要 的 是 计算 自己 的 报税 数据 那么 他 所 需要 软件 的 用 
途 就 是 帮助 用 户 计算 报税 数据 











SSA 
户 要 做 正事 的 时 候 ， 当 面 弹出 几 个 窗口 ， 或 者 新 增 一 堆 功 能 ， 让 他 无 
法 分 辨 ， 或 者 显示 一 大 堆 他 完全 不 认识 的 图 标 。 让 用 户 勃 然 大 把 方法 
多 多 ， 都 无 非 是 干扰 用 户 的 需求 ， 背 离 软件 自身 的 基本 用 途 。 


有 时 候 ， 营 销 人 员 或 者 经 理会 给 软件 设 定 一 些 目标 ， 但 是 这 些 目标 其 
实 并 不 符合 程序 的 基本 用 途 ， 比 如 “要 好 玩 一些 ".“ 设 计 要 更 有 冲击 
感 "、“ 要 受 新 媒体 欢迎 ",“ 要 使 用 最 新 的 技术 ”等 等 。 这 些 人 可 能 是 
公司 里 的 重要 人 物 ， 但 他 们 不 是 那些 决定 程序 应 当做 什么 的 人 。 身 为 
软件 设计 师 或 是 技术 经 理 ， 你 的 职责 是 保证 软件 的 基本 用 途 ， 防 止 它 
偏离 正轨 ， 这 个 责任 其 他 人 谁 也 担 不 起 。 有 时 候 ， 你 可 能 需要 为 此 据 
理 力争 ， 但 从 长 期 来 看 ， 这 肯定 是 值得 的 。 























这 古寺 真 万 确 的 ， 设 有 必要 玩 太 多 花样， 做 太 复杂 ， 芝 试用 单 小 软件 
瞬间 完成 500 MES. EAN PARE SEM elie Ay, FEA 
ANZTZAHR. 
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8.2 ”糟糕 的 技术 


出 现 复杂 性 的 另 一 个 常见 原因 就 是 ， 系 统 里 选择 了 错误 的 技术 ， 尤 其 
是 最 终 发 现 并 不 能 很 好 适应 未 来 需求 的 技术 。 但 是 既然 无 法 观测 未 
来 ,现在 就 决定 要 选择 什么 技术 并 不 简单 。 好 在 ， 开始 使 用 之 前 ， 你 
可 以 通过 三 个 因素 来 判断 技术 是 否 “ 糟 糕 ": 生存 潜力、 互通 性 、 对 
品质 的 重视 。 


8.2.1 生存 潜力 


某 种 技术 的 生存 玉 力 ， 就 是 它 持续 获得 维护 的 可 能 性 。 如 果 茶 种 类 库 
或 依赖 项 已 经 过 时 ， 没 有 人 维护 ， 你 却 死 守 着 它们 ， 就 有 麻烦 在 等 着 
你 了 。 


要 了 解 某 款 软件 的 生存 玖 力 ， 可 以 查阅 其 最 近 的 发 布 记录 。 开 发 者 是 
否 频 营地 发 布 真 正解 决 了 客户 问题 的 新 版 本 ? 田 外 ， 开 发 者 对 bug th 
告 的 响应 有 多 积极 ?他们 是 否 有 活跃 的 邮件 列表 或 支持 团队 ? EGA 
足够 的 人 在 线 谈论 这 项 技术 ?如 果菜 项 技术 现在 有 是 够 的 活力 ， 你 可 
LATS Ae EASA KT. 


FIM AA, ER AARNE ERA A, UR TABA 
同 开发 人 员 开 发 的 各 种 程序 所 接受 和 使 用 。 如 采 只 有 一 家 供应 商 来 推动 
和 改造 ， 那 么 万 一 供应 商 甩 手 不 管 或 者 决定 停止 维护 ， 你 就 麻烦 了 。 


接受 广泛 程度 
这 个 说 法 听 起 来 的 意思 似乎 是 ， 应 当 从 和 衬 合 需求 的 技术 中 选择 接 
受 最 广泛 的 。 从 某 种 程度 上 说 这 是 真 的 ， 被 接受 的 技术 一 般 都 是 有 相 
当 大 的 生存 法力。 不过， 你 还 是 ,必须 辨别 ， 到 底 是 经 过 者 验 才 被 大 家 
ER, LARRLARDAKLH ER ORES, 


在 写作 本 书 时 ，C 是 经 过 验证 的 广泛 接受 的 语言 。 在 各 种 公司 ， 
有 各 种 人 ， 为 各 种 不 同 的 目的 使 用 CC 语言 。 关 于 C 语 言 有 许多 国际 标 
准 ， 每 项 标准 又 有 众多 的 实现 ， 包 含 许 多 三 汉 使 用 的 编译 器 。 
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不 过 ， 有 些 技术 之 所 以 被 广泛 接受 ， 是 因为 用 户 别 无 选择 :。 假 
设 XX 公司 设计 了 自己 的 编程 语言 ， 然 后 又 设计 了 一 种 广泛 接受 的 设 
备 ， 这 种 设备 只 接受 这 种 语言 编写 的 程序 。 这 就 是 上 面 提 到 的 “单一 


供应 商 ” 情况， 语言 看 起 来 被 广泛 接受 ， 但 生存 潜力 其 实 很 小 ， 除 非 
它 能 被 整个 软件 行业 广泛 接受 。 





822 互通 性 


所 硼 互 通 性 ， 指 的 是 如 末 和 需要， 从 一 种 技术 切换 到 男 一 种 技术 有 多 
难 。 要 了 解 技术 的 互通 性 ， 就 得 问 问 自己 :“ 我 们 能 不 能 以 某 种 标准 
方式 来 交互 ， 这 样 就 更 容易 切换 到 遵循 同样 标准 的 其 他 系统 ? ” 


举例 来 说 ， 有 些 国际 标准 规定 了 程序 应 当 如 何 与 数据 库 系统 交互 。 有 
些 数据 库 对 这 种 标准 的 支持 很 好 ， 如 果 你 选择 这 些 支 持 良好 的 数据 
库 ， 将 来 程序 只 需要 做 很 小 的 改动 ， 就 很 容易 切换 到 其 他 数据 库 。 


不 过 ， 也 有 些 数据 库 对 标准 的 支持 并 不 好 。 如 果 你 希望 切换 到 这 种 支 
持 精 粽 肝 数据库 ， 束 得 重 写 自 己 的 程序 。 所 以 ， 如 果 选 择 非 标准 系 
统 ， 就 会 身 陷 罗网 ， 难 以 切换 。 


8.2.3 对 品质 的 重视 


这 十 一 种 更 主观 的 衡量 ， 其 思想 十 考察 最 近 的 发 布 中 ， 产 品 是 否 更 加 
完善 了。 如 琳 你 可 以 看 到 产 人 代码， 检查 一 下 开发 人 员 十 人 否 进 行 了 重 
构 ， 清 理 了 代码 。 产 品 是 更 容易 使 用 了 ， 还 是 更 难 用 了 ? 维护 它 的 人 
真 的 在 乎 产品 的 质量 吗 ? 最 近 是 否 多 次 出 现 似 乎 由 精 糕 的 编程 所 引发 
的 严重 安全 问题 ? 


El: 开发 人 员 可 能 对 目 己 所 用 的 技术 有 独特 的 热情 。 为 了 避免 触犯 这 类 用 尸 ， 我 们 还 是 
不 提 具 体 的 技术 为 佳 。 
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824 其 他 原因 


在 选择 技术 时 ， 还 有 其 他 因素 需要 考虑 ， 主 要 是 它 是 否 人 简洁 ， 是 否 符 
合 软件 的 基本 用 途 。 在 衡量 完 所 有 实际 因素 之 后 ， 也 可 以 考虑 个 人 喜 
好 。 有 些 人 更 看 重 东 种 编程 语言 在 某 些 方面 的 优势 。 有 时 候 ， 这 也 古 
选择 技术 的 有 效 理 由 : 如 采 两 种 技术 在 其 他 方面 都 老 不 多 ， 孢 会 选择 
更 喜欢 的 那 种 。 毕 竞 ， 节 终 真正 使 用 它 的 人 是 你 ， 你 的 意见 很 重要 ! 
上 面 这 些 指 引 只 负 和 贡 帮 你 排除 掉 粳 糕 的 选择 ， 剩 下 的 则 取决 于 你 自己 
的 研究 、 需 求 、 计 划 。 


8.3 复杂 性 及 销 误 的 解决 方案 


通常 ， 如 末 某 件 事情 变 得 非常 复杂 ， 也 就 意味 绝 不 是 表面 的 复杂 那么 
A, MERA Sa, 


Ie ANY AI HE Te TBR, EIRAHFTR. IR, ERA 
能 解决 问题 ， 你 需要 重新 设计 汽车 ， 换 上 圆 形 的 轮子 。 


一 旦 程序 里 出 现 了 “无 法 解决 的 复杂 性 ”"， 就 说 明 设 计 中 有 些 深 层次 
的 基本 错误 。 如 采 问 题 在 这 个 层面 上 无 法 解决 ， 应 当 回 过 头 去 看 看 产 
AE AAA BIER AE TT A 


其 实 ， 程 序 员 经 币 这 么 做 。 你 可 能 会 说 : 这 代码 太 垃 圾 了， 要 添加 新 
功能 真 够 嫉 烦 的 。 那 么 ， 次 层次 的 基本 问题 就 是 代码 太 混 乱 。 所 以 ， 
你 应 该 丈 理 这 些 代 码 ， 让 和 它 们 变 和 商洛 ， 你 就 会 发 现 增加 新 功能 也 会 变 
HE, 


真正 要 解决 的 问题 是 什么 ? 


如 采 有 人 问 你 : “怎么 让 小 马 飞 上 月 球 ? ”你 要 反问 的 就 是 :“ 要 解决 
的 完 竟 是 什么 问题 ? ”你 可 能 发 现 ， 这 个 人 真正 需要 的 其 实 是 收集 一 
些 灰 色 的 岩石 。 为 什么 要 去 月 球 ， 还 必须 弄 一 匹 小 马 ， 只 有 他 自己 知 
道 。 很 多 人 真 的 会 弄 混 清 这 类 问题 ， 所 以 应 当 问 问 他 们 ， 究 竟 要 解决 
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什么 问题 ， 然 后 简单 的 办 法 就 会 自己 冒 出 来 。 在 上 面 的 例子 里 ， 一 且 
彻底 弄 清 楚 了 问题 ， 解 决 方案 就 很 简单 直接 ,去 室外 找 些 灰色 的 岩石 
就 够 了 ， 根 本 用 不 着 小 马 。 


所 以 ， 如 果 事 情 变 复杂 ， 不 妨 回 过 头 去 看 看 真正 要 解决 的 是 什么 问 
题 。 你 可 以 退 上 一 大 步 ， 也 可 以 质疑 任何 问题 。 有 可 能 为 了 得 到 A, 
你 想到 的 是 2+2， 但 没 想到 1+3 也 可 以 ， 或 者 干脆 不 要 加 法 ， 直 接 取 
得 4 也 行 。 真 正 的 问题 是 “我 如 何 得 到 4 这 个 数字 "， 任 何 可 行 的 广 
案 都 是 可 以 接受 的 。 所 以 你 真正 要 做 的 就 是 ， 找 出 自己 所 处 的 环境 中 
最 好 的 办 法 。 


要 做 到 这 一 点 ， 不 应 当 依 徘 猜 副 ， 而 必须 亲眼 去 看 要 解决 的 问题 。 确 
认 你 真正 理解 了 问题 的 方方面面 ， 我 到 最 简单 的 解决 办 法 。 不 要 问 
用 现 有 代码 怎么 解决 这 个 同 题 "， 或 者 “Anne 教授 在 程序 里 是 怎么 
解决 这 个 问题 的 ， 而 是 问 同 你 自己 :“ 通 和 营 情 况 下 ， 在 最 完美 的 方案 
里 ， 这 业 辣 题 要 如 何 解决 ? ”这 样 ， 你 大 概 就 可 以 看 出 代码 应 该 如 何 
重 写 了 。 然 后 ， 就 可 以 着 手 解 决 问题 。 


然后 ， 冲 题 就 不 复 存在 了 。 


8.4 复杂 问题 

有 时 候 你 受命 去 解决 一 些 本 身 就 非常 复杂 的 问题 ， 比 如 编写 拼写 检查 
或 国际 象棋 的 程序 。 问 题 复杂 ， 解 法 不 一 定 会 复杂 ; 相反 ， 在 处 理 此 
基 癌 题 了 时， 你 必须 更 努力 地 过 求 代码 的 倍 社 。 

如 果 你 在 解决 复杂 问题 时 遇 到 了 有 麻烦， 那么 用 简单 易 懂 的 文字 把 它 写 
在 纸 上 ， 或 者 画 出 来 。 有些 最 优秀 的 程序 设计 可 是 在 纸 上 完 成 的 ， 真 
的 。 把 它 输入 到 计算 机 里 只 是 次 要 的 细节 。 


大 多 数 麻 频 的 设计 问题 ， 都 可 以 用 在 纸 上 画 图 或 写 出 来 的 
办 法 找到 答案 。 
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8.5 WITZ BETS 


身 为 程序 员 ， 你 必然 要 面 对 复 末 性 。 其 他 程序 员 会 写 复杂 有 的 程序 ， 你 
必须 去 修正 。 硬 件 设计 师 和 语言 设计 师 会 让 你 的 生活 更 用 和 烦 。 


IN 73 


如 果 系 统 中 某 个 部 分 太 过 复杂 ， 有 个 好 办 法 来 解决 : 把 它 分 解 成 几 个 
独立 的 小 部 分 ， 逐 步 重新 设计 。 每 次 修改 都 应 该 足够 小 ， 这 样 可 以 放 
LOAF, BRUSHES A. MM PURA RAN Eke, FF 
修改 有 可 能 会 引 人 更 多 的 复杂 性 。 许 多 重 设 计 或 重 写 的 工作 之 所 以 草 
终 失 败 ， 束 是 因为 它们 引 人 了 更 多 的 复杂 性 ， 结 条 最 后 和 原 有 系统 同 
RRR 


每 小 步骤 都 应 该 足够 小 ， 比 如 给 某 个 变量 取 个 更 好 的 名 字 ， 或 是 给 难 
懂 的 代码 增加 一 些 注释 。 更 带 见 的 做 法 是 在 每 个 步骤 中 都 把 一 个 复 
杂 的 部 分 拆 分 成 若干 个 简单 的 部 分 。 


如 汪 所 有 的 代码 都 包含 在 一 个 巨大 的 文件 里 ， 改 进 的 第 一 步 束 是 把 茶 
个 部 分 保存 到 单独 的 文件 里 。 之 后 改进 这 个 小 部 分 有 的 设计 ， 然 后 把 男 
一 个 部 分 保存 到 新 的 文件 ， 再 改进 这 个 部 分 的 设计 。 如 此 重复 下 去 ， 
最 终 得 到 的 就 是 可 靠 的 、 可 理解 的 、 可 维护 的 系统 。 


如 果 系 统 非常 复杂 ， 这 么 做 的 工作 量 可 能 相当 大 ， 所 以 必须 有 耐心 。 
你 必须 首先 做 好 设计 ， 改 进 之 后 的 系统 要 比 现在 的 简单 一 即便 只 是 
简单 一 点 点 。 然 后 ， 朝 着 这 个 目标 一 步 步 地 前 进 。 得 到 了 这 个 简单 的 
系统 之 后 ， 就 设计 下 一 个 更 简单 的 系统 ， 再 朝 那个 目标 前 进 。 不 必 i 
计 “ 完 美 ”的 系统 ， 因 为 它 不 存在 。 你 只 需要 持续 不 懈 地 追求 比 : 
系统 更 好 的 系统 ， 最 终 就 会 得 到 相当 容易 管理 的 简洁 系统 。 


过 ， 还 ， ARES, A 停 

新 功能 。 变 化 定律 告 i ， 程 序 所 处 的 环境 是 持续 变化 的 ， 
所 以 程序 的 功能 也 必须 去 适应 这 些 变化 。 如 果 在 相当 长 的 时 间 里 ， 你 
都 不 能 从 用 户 角度 出 发 来 调适 和 改进 ， 就 可 能 失去 自己 的 用 户 ， 把 项 
AS. 
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好 在 平衡 开发 新 功能 和 应 对 复杂 性 这 两 项 任务 的 方法 有 很 多 。 最 好 的 
一 个 办 法 就 症 ， 重 新 设计 时 只 卷 虑 让 新 功能 更 容 多 实现， 然后 实现 这 
个 功能 。 这 样 ， 你 就 可 以 在 重新 设计 和 开发 新 功能 之 间 定 期 切换 。 它 
同样 有 助 于 保证 新 设计 能 够 适应 需求 ， 因 为 设计 时 会 考虑 到 实际 的 应 
用 。 系 统 的 复杂 性 也 会 逐渐 下 降 ， 而 且 你 一 直 都 跟 得 上 用 户 的 需求 。 
你 甚至 可 以 这 样 来 处 理 bug: 如 果 发 现 修 改 设计 之 后 ， 茶 些 bug ER 
易 修 复 ， 那 么 先 重新 设计 代码 再 修复 bug, 


为 新 增 功 能 重新 设计 


有 个 叫 Bugzilla 的 项 目 ， 它 的 所 有 数据 都 存储 在 数据 库 中 。Bugzilla 
只 支持 用 某 种 特定 的 数据 库 来 存储 ， 这 里 我 们 叫 它 OldDB。 有 些 新 
客户 要 求 使 用 别 的 数据 库 来 存储 数据 ， 我 们 叫 它 NewDB。 客 户 的 理 
由 很 充分 : 相 比 OldDB， 他 们 更 熟悉 NewDB， 而 且 他 们 公司 已 经 在 
用 NewDB 了 。 但 是 ， 原 有 的 客户 布 望 继 续 使 用 DldDB。 


所 以 ，Bugzilla 必 须 支 持 多 种 数据 库 。 这 需要 大 量 修改 代码 ， 因 为 
Bugzilla 没 有 统一 存 取 数 据 库 的 接口 。 相 反 ， 代 码 里 散落 着 很 多 自 定 
义 的 数据 库 处 理 指令 ， 它 们 只 适用 于 OIdDB， 而 无 法 对 接 NewDB，。 
一 个 解决 办 法 是 在 代码 中 增加 许多 if 语 和 铝 ， 在 每 个 访问 数据 库 的 地 
方 区 分 NewDB 和 OI1dDB 特 殊 处 理 。 这 样 代码 库 的 复杂 性 几乎 会 加 
倍 ， 而 Bugzilla 的 团队 只 有 少数 兼职 程序 员 ， 要 是 系统 的 复杂 性 加 
倍 ， 他 们 就 维护 不 了 了 。 


于 是 ，Bugzilla 团 队 决 定 采 取 另 一 个 办 法 ， 重 新 设计 系统 ， 让 它 能 容 
易 支 持 多 种 数据 库 。 这 样 的 改动 是 个 大 工程 。 以 下 概要 描述 了 他 们 
的 步骤 。 


(1) Bugzilla 中 对 所 有 数据 库 系 统 已 经 有 一 套 操 作 标 准 ， 但 没有 广泛 
使 用 。 所 以 检查 整个 系统 ， 每 次 修改 一 个 文件 ， 只 要 可 能 ， 就 把 它 
切换 为 标准 操作 。 

(2) 对 没有 标准 版 本 的 数据 库 操 作 ， 写 一 个 浮 数 ， 它 会 返回 适用 当前 
所 用 数据 库 的 正确 操作 。 为 每 个 非 标准 操作 创建 一 个 函数 ， 把 非 标准 
操作 替换 为 函数 调用 。 重 复 这 个 过 程 ， 直 到 去 掉 所 有 的 非 标 准 函 数 。 
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3) 项 目 中 有 许多 代码 服务 于 专属 于 OldDB 的 功能 。 停 止 使 用 这 些 
OldDB 专 有 的 功能 ， 切 换 到 可 适用 于 所 有 数据 库 的 标准 功能 。 每 次 
只 修改 一 个 功能 ， 如 果 需 要 ， 还 可 以 分 更 细 的 步骤 。 

(4) 重新 设计 Bugziila 的 安装 程序 ， 让 它 可 以 设 定 任何 数据 库 系 统 ， 
而 不 只 是 OldDB。 首 先 要 重新 设计 安装 程序 ， 把 它 变 简单 ， 然 后 整 
理 代码 ， 使 其 同时 支持 ODIdDB 和 NewDB。 

以 上 任何 一 步 本 身 都 是 一 个 项 目 。 它 们 被 拆 分 为 小 的 步骤 ， 所 以 每 
一 步 的 工作 都 得 到 了 良好 的 设计 。 而 有 全， 系统 做 了 任何 改动 之 后 部 
会 进行 测试 ， 确 保 对 OldDB 的 功能 没有 变化 。 


这 样 最 后 得 到 了 一 个 完美 的 系统 吗 ? 没有 ， 只 是 得 到 了 比 之 前 更 好 
的 系统 一 一 不 但 支持 NewDB ， 而 且 更 容易 维护 。 最 终 Bugzilla 能 够 
支持 4 种 不 同 的 数据 库 ， 这 全 都 是 因为 做 了 上 面 的 工作 后 ， 它 更 容 
易 支 持 新 的 数据 库 了 。” 





8.5.1 把 某 个 部 分 变 简单 


上 面 说 的 都 很 不 错 ， 很 有 道理 ， 但 到 底 该 做 什么 才能 把 某 个 部 分 变 简 
单 ? 咽 ， 这 就 要 用 到 现 有 的 关于 软件 设计 的 知识 了 。 学 习 设 计 模式 和 
设计 方法 来 处 理 遗 留 代 码 ， 学 习 软 件 工程 师 普 过 使 用 的 工具 ， 都 可 以 
玫 上 大 和 低 ， 尤 其 有 用 的 古 擎 握 多 门 编程 语言 ， 熟 悉 多 种 不 同 的 类 库 ， 
因为 每 一 种 语言 和 类 库 都 会 用 自己 的 方式 来 思考 问题 ， 即 便 你 并 不 使 
用 这 些 语 言 和 类 库 ， 这 种 思维 方式 也 可 以 应 用 到 你 的 具体 环境 中 。 


掌握 了 这 些 知识 ， 在 面 对 复 杂 性 问题 时 你 就 会 有 更 多 的 选择 。 软 件 设 
计 的 法 则 可 以 帮助 你 选择 不 错 的 选项 ， 在 这 之 后 ， 你 的 判断 力 和 经 验 


注 2; 在 过 去 许多 年 间 ， 因 为 各 种 原因 ，Bugzilla 按照 这 个 套路 重新 设计 了 很 多 次 。 如 
果 你 想 知 道 已 经 完成 的 重要 工作 ， 可 以 查阅 这 里 已 经 划 掉 的 项 : https:Wbugzilla. 
mozilla.org/showdependencytree.cgi?id=278579&hide_resolved=0, 4032 (R78 Mig KF 
数据 库 的 改动 如 何 完成 的 更 详细 情况 ， 请 查阅 这 里 已 经 划 掉 的 项 :https:Wbugzilla. 
mozilla.org/showdependencytree.cgi?id=98304&hide_resolved=0。 如 果 你 熟悉 数据 库 
系统 ， 阅 读 各 项 的 标题 就 大 概 知道 整个 项 目 是 怎么 完成 的 了 。 
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可 以 决定 针对 当前 的 具体 问题 要 做 什么 。 不 要 仅仅 因为 权威 的 肯定 就 
机 械 地 生 搬 硬 套 某 个 工具 ， 我 们 的 选择 永远 是 ， 在 当前 的 环境 下 ， 当 


前 的 代码 中 ， 做 合适 的 事情 。 


虽然 有 时 会 有 这 样 的 情况 : 你 看 到 某 一 段 代 码 ， 却 不 知道 有 什么 工具 
可 以 化 出 它 。 或 者 你 可 能 刚刚 开始 学 编程 ， 还 没有 时 间 马 上 研究 这 些 
信息 。 如 果 是 这 样 ， 你 应 该 看 着 这 个 复杂 问题 问 自 己 : 要 怎么 做 ， 才 
可 以 让 事情 处 理 或 是 理解 起 来 更 容易 ? 这 就 是 在 每 个 化 解 复杂 性 的 问 
题 育 后 的 关键 应 。 应 对 之 道 在 于 找到 一 种 让 代码 更 简单 的 可 行 办 法 。 
软件 设计 的 工具 和 技巧 只 能 锦上添花 ， 帮 助 你 找到 更 好 的 答案 。 




















8.5.2 不 可 解决 的 复杂 性 
RSA, Ay RE AT CE FAY 


简化 系统 时 ， 你 可 能 会 发 现 某 些 复杂 性 是 无 可 过 
硬件 就 是 很 复杂 的 。 如 果 过 到 这 类 不 可 解决 的 复杂 性 ， 你 要 做 的 就 是 
屏 节 这 种 复杂 性 。 在 程序 外 面 妥 善 包 装 上 一 层 ， 让 其 他 和 请 














面 对 非 常 复杂 的 系统 ， 有 些 设计 师 会 选择 推倒 重 来。 
到 重 来 ， 几 乎 就 是 设计 师承 认 失 败 ， 等 于 说 :“: 
的 系统 ， 所 以 只 能 重 来 。 


有 人 认为 所 有 的 系统 最 终 都 要 重 写 ， 但 事实 不 是 这 样 的。 系统 在 设计 
完成 之 后 永远 不 会 被 拢 弃 是 有 可 能 做 到 的 。 软 件 设计 师 所 说 的 “ 某 
天 ， 因 为 某 个 原因 ， 我 们 不 得 不 抛弃 所 有 代码 "， 就 好 像 建筑 师 说 的 
“这 座 摩 天 大 楼 应 该 在 某 天 以 某 种 方式 倒塌 "。 如 果 摩 天 大 楼 设计 有 问 
题 ， 后 期 也 没有 很 好 地 维护 ， 那 么 它 必然 会 倒塌 。 可 要 是 从 一 开始 就 
建 得 很 好 ， 而 且 维 护 得 很 好 ， 为 什么 会 倒塌 呢 ? 


既然 能 建 但 出 坚固 稳定 的 摩天 大 楼 ， 也 就 能 构建 出 可 维护 的 软件 系统 。 
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上 面 说 了 这 么 多 ， 不 过 ， 还 是 有 些 时候 重 写 是 可 以 接受 的 。 但 是 ， 这 
种 情况 非常 罕见 。 如 果 下 面 的 条 件 全 都 福 足 ， 你 才 应 该 重 与 。 


(1) 你 已 经 完成 了 准确 评估 ， 证 明 重 写 整个 系统 会 比重 新 设计 现 有 系 
统 更 有 效率 。 只 有 猜测 是 不 够 的 ， 你 需要 真正 去 做 一 些 重新 设计 现 有 
系统 的 试验 ， 然 后 对 比 结 果 。 已 有 的 复杂 系统 可 能 很 难 应 付 ， 某 些 部 
分 可 能 很 难处 理 ， 但 是 为 了 知道 修复 它 需 要 多 少时 间 ， 你 必须 动手 做 
一 些 尝试 ，。 

(2) 你 有 足够 的 时 间 用 来 开发 新 的 系统 ， 

(3) 你 要 比 原 有 系统 的 设计 师 更 高 明 ， 或 者 ， 如 果 原 有 系统 古 你 设计 
的 ， 但 现在 你 的 设计 能 力 已 经 大 大 提升 了 。 

(4) 你 完全 打算 好 了 通过 一 系列 简单 的 步骤 设计 整个 新 系统 ， 在 每 一 
步 都 有 用 户 提供 反馈 。 | 

(5) 你 有 足够 的 资源 ， 可 兼顾 维护 原 有 系统 和 重新 设计 系统 。 绝 对 不 
要 为 了 让 程序 员 重 写 新 系统 而 停止 对 原 有 系统 的 维护 。 系 统 只 要 在 使 
用 ， 都 离 不 开 维 护 。 请 记 住 ， 你 自己 的 精力 也 是 一 种 资源 ， 必 须 慎重 
EME MAA ER, MERA SRA AAA A 
统 吗 ? 


如 果 上 面 的 条 件 都 注 足 ， 那 么 推倒 重 来 是 可 以 接受 的 。 否 则 ， 应 读 做 
的 事情 不 是 推倒 重 来 ， 而 是 降低 现 有 系统 的 复杂 性 ， 也 就 是 通过 一 系 
列 简单 步骤 来 改进 系统 的 设计 。 
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我 们 无 法 保证 程序 将 来 一 直 可 以 正常 使 用 ， 只 能 保证 目前 它 可 以 正常 
运行 。 甚 至 这 次 运行 是 正常 的 ， 而 下 次 就 可 能 出 现 问题 。 可 能 周围 的 
环境 发 生 了 变化 ， 所 以 程序 无 法 运行 ， 也 可 能 你 换 了 一 台 机 器 ， 所 以 
无 法 运行 。 


不 过 ， 项 望 还 是 有 的 一 一 我 们 并 不 会 备 受 程 序 功能 不 确定 的 无 穷 煎熬 ， 
测试 法 则 (Law of Testing) 告诉 我 们 : 
你 对 软件 行为 的 了 解 程度 ， 等 于 你 真正 测试 它 的 柱 度 。 

软件 最 后 一 次 测试 的 时 间距 离 现 在 越 近 ， 它 可 以 继续 正常 运行 的 可 能 
性 就 越 大 。 软 件 在 越 多 的 环境 里 测试 过 ， 你 就 越 确 定 它 可 以 运行 在 这 
些 环境 中 。 上 面 所 说 的 测试 的 “程度 ”"， 包 括 软 件 有 多 少 个 方面 你 曾 
经 油 起 过， 上 一 次 镜 试 古 多 和 以前， 在 多 少 不 同 的 环境 下 进行 过 宰 
试 。 总 的 来 说 ， 你 可 以 这 么 理解 


除非 来 自 测试 过 ， 否 则 你 不 知道 软件 是 否 能 正常 运行 。 


仅仅 “能 正常 运行 "， 其 实 是 非常 模糊 的 ， 所 请 的 “能 正常 运行 ”是 
指 什 么 呢 ? 你 在 测试 时 真正 知道 的 是 ， 软 件 的 行为 是 否 符 合 预期 ， 所 
以 ， 你 必须 清楚 地 知道 预期 的 行为 。 这 个 要 求 看 来 有 点 傻 ， 而 且 是 理 
所 当然 的 ， 但 是 测试 的 关键 就 在 于 此 。 针 对 每 项 油 试 ， 你 必须 癌 一 个 
非常 具体 的 问题 ， 得 到 一 个 非常 具体 的 若 案 。 问 题 可 能 征 :程序 局 
动 之 后 ， 用 户 会 首先 看 到 这 个 按钮 ， 那 么 程序 第 一 次 运行 的 时 候 ， 
ARIAT Pix TRA, SRATA? ”然后 你 会 得 到 具体 的 答案 ， 比 
如 :;“ 程 序 会 弹出 一 个 窗口 ， 显 示 “Hello, World 。 

这 样 ， 就 有 了 一 个 问题 ， 而 且 知 道 了 答案 。 如 果 还 有 其 他 的 答案 ， 那 
么 软件 就 是 “不 能 正常 运行 ”的 。 

AE AEREA (ARE: “MRAP RAR, BAS Ait 
吗 ? ”期 望 回 答 是 “不 会 "。 但 是 对 设计 良好 的 软件 ， 在 大 多 数 情况 
下 ， 出 试 之 后 你 会 得 到 具体 得 多 的 信息 。 
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你 还 必须 保证 测试 是 准确 的 。 如 果 测 试 显示 程序 的 行为 完 
4 。 全 符合 预期 ， 事 实 却 并 非 如 此 ， 或 者 测试 显示 程序 崩溃 了 ， 
… 其 实 它 跑 得 好 好 的 ， 那 么 这 些 测试 都 是 不 准确 的 。 





最 后 ， 你 必须 观 宗 铀 试 的 结 有 未， 确保 它 们 是 有 效 的 。 如 全副 试 失败 
Ly, DARME FAA MAL, UM AMADA A IRA. 


WAT REAL AA AE MBA. RAIS ERA AGA SS, AS 
记 了 看 它 十 否 能 正 带 运行 。 可 十 ， 无 论 程序 员 多 么 聪明 ， 也 无 论 有 多 
少 理论 数据 来 证 明 你 的 代码 是 正确 的 ， 在 没有 切实 测试 过 之 前 ， 你 都 
不 知 坦 它 能 不 能 正常 工作 ， 


只 要 你 修改 了 软件 的 某 个 部 分 ， 这 部 分 是 否 能 正常 工作 就 成 了 未 知 
数 ， 就 必须 重新 测试 。 而 且 ， 这 部 分 很 可 能 关联 到 其 他 许多 方面 ， 所 
以 你 不 知道 关联 的 部 分 是 否 会 受 影 响 。 如 果 是 大 刀 阔 人 莽 地 改 ， 你 可 能 
就 需要 重新 测试 整个 程序 。 


显然 ， 你 不 会 希望 每 次 进行 了 一 点 小 修改 之 后 ， 还 要 手工 测试 整个 程 
序 。 所 以 如 今 的 程序 员 在 落实 测试 法 则 时 ， 会 编写 许多 日 动 化 测试 来 
测试 所 写 的 每 一 段 代码 。 这 样 做 的 好 处 在 于 ， 有 任何 改动 都 只 需要 重 
新 运行 这 些 初试 即 可 ， 程 序 会 测试 系统 中 的 方方面面 ， 确 保修 改 之 后 
所 有 部 分 仍然 保持 正常 。 


网 上 有 许多 轩 料 讲解 如 何 编 写 自动 化 测试 以 及 测试 的 一 般 原 理 ， 也 出 
版 过 很 多 相关 书 藉 。 测 试 领域 的 资料 丰富 详尽 ， 值 得 学 习 。 测 试 法 则 
只 负责 解释 为 什么 需要 测试 ， 什 么 时 候 需 要 测试 ， 以 及 测试 真正 能 提 
供 什么 信息 。 
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ARETE AS BRENNT TIER], 


(1) 软件 的 目的 是 帮助 他 人 。 

(2) 软件 充 计 的 方程 式 是 : 
_ Y +W 

Ei T Em 

其 中 ， 

DPV A REE CATE); 

V, 表示 未 来 价值 ; 

É, 表示 开发 成 本 ; 

En 表示 维护 成 本 。 

这 是 软件 设计 的 主要 法 则 。 随 时 间 的 推移 ， 这 个 方程 式 简化 为 : 
p= 

Em 


也 就 是 说 ， 相 比 降低 实现 成 本 ， 降 低 维 护 成 本 更 重要 。 

(3) 变化 定律 : 程序 存在 的 时 间 越 久 ， 它 的 某 个 部 分 需要 变化 的 可 能 
HERA, 

Meer ACRE RARA A a RE PES ARE RELL. 

(5) 简 清 定律 ， 软件 任何 一 部 分 的 维护 难度 ， 反 比 于 该 部 分 的 简洁 程度 。 
(6) META: 你 对 软件 行为 的 了 解 程度 ， 等 于 你 真正 而 试 它 的 程度 。 
就 是 这 么 多 。 本 书 中 讨论 了 很 多 的 事实 和 想法 ， 但 这 6 条 是 软件 设计 
的 法 则 。 请 注意 ， 其 中 最 重要 的 是 要 牢记 软件 的 目的 、 软 件 设 计 方程 
式 的 简化 形式 以 及 简洁 定律 。 

如 果 你 希望 把 最 重要 的 事实 综合 成 软件 设计 时 要 记得 的 两 句 话 ， 就 是 ， 
。 相 比 降低 开发 成 本 ， 降 低 维护 成 本 更 加 重要 ， 

。 维护 成 本 正比 于 系统 的 复杂 程度 。 

有 了 这 了 两 条 ， 以 及 对 软件 目的 的 了 解 ， 再 加 上 你 知道 整个 系统 的 复杂 
性 源 目 各 部 分 的 复杂 性 ， 你 就 有 相当 的 把 握 去 重新 认识 软件 设计 这 整 
门 科 学 。 
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